From 3b49cd1c7916fc87024e9fa401dd6783e8f32e89 Mon Sep 17 00:00:00 2001 From: LexLuthr <88259624+LexLuthr@users.noreply.github.com> Date: Mon, 19 Feb 2024 16:47:48 +0530 Subject: [PATCH] fix: remove boost-gfm and go-data-transfer (#1740) * remove gfm, pull datatransfer * make gen, docsgen * fix circular depndencies * go mod tidy * fix lint errors * convert RetrievalAskGetter to interface * fix interface pointer * fix dependency injection, move ask db * fix itests * clean up mocks * chore: lots more dt & gs debugging (#1776) * fix lint errs, add back ProviderDataTransfer * regenrate cbor * cleanup go-data-transfer * fix graphsync retrievals * use new client for GS tests * fix selector type * restore selector type * fix: clean up and fix retrieval framework & tests (#1848) * fix: clean up and fix retrieval framework & tests * Use new go-trustless-utils primitives and traversal tools * Simplify Framework#Retrieve * fix: wait for full startup of data transfer when making client * fix lint errors * fix go mod * cleanup itest fixtures, fix gql * go mod tidy * fix directDealProv * handle storage ask * cleanup config * remove graphsync unit tests * add retrieval config headers * update retrieval configs * fix storage ask db command * fix BoostLegacyDealByProposalCid api * make cbor-gen * fix missing api example * migrate config, dump dagstore config --------- Co-authored-by: Rod Vagg --- .circleci/config.yml | 27 +- api/api.go | 90 +- api/api_full.go | 80 +- api/docgen/docgen.go | 25 +- api/proxy_gen.go | 507 +-- api/types.go | 66 - build/openrpc/boost.json.gz | Bin 5845 -> 3126 bytes cmd/boost/provider_cmd.go | 100 +- cmd/boostd/dagstore.go | 357 -- cmd/boostd/import_data.go | 14 +- cmd/boostd/index.go | 8 +- cmd/boostd/init.go | 521 --- cmd/boostd/legacy_data_transfers.go | 325 -- cmd/boostd/legacy_retrieval_deals.go | 277 -- cmd/boostd/main.go | 6 - cmd/boostd/piecedir.go | 1 + cmd/boostd/recover.go | 2 +- cmd/booster-http/e2e_test.go | 2 +- cmd/booster-http/piecehandler.go | 4 +- cmd/booster-http/trustless_gateway_test.go | 2 +- cmd/boostx/utils_cmd.go | 2 +- cmd/lib/common.go | 10 +- cmd/lib/stores/dagstore.go | 12 + cmd/lib/stores/error.go | 11 + cmd/lib/stores/filestore.go | 164 + cmd/lib/stores/kvcarbs.go | 1676 +++++++++ cmd/lib/stores/ro_bstores.go | 60 + cmd/lib/stores/rw_bstores.go | 63 + cmd/migrate-lid/migrate_lid.go | 6 +- datatransfer/channelmonitor/channelmonitor.go | 490 +++ datatransfer/channels/block_index_cache.go | 63 + datatransfer/channels/channel_state.go | 251 ++ datatransfer/channels/channels.go | 414 +++ datatransfer/channels/channels_fsm.go | 298 ++ .../channels/internal/internalchannel.go | 90 + .../internal/internalchannel_cbor_gen.go | 1083 ++++++ .../internal/migrations/migrations.go | 13 + datatransfer/encoding/encoding.go | 171 + datatransfer/errors.go | 32 + datatransfer/events.go | 160 + datatransfer/impl/environment.go | 27 + datatransfer/impl/events.go | 590 +++ datatransfer/impl/impl.go | 561 +++ datatransfer/impl/receiver.go | 192 + datatransfer/impl/restart.go | 198 + datatransfer/impl/timecounter.go | 21 + datatransfer/impl/utils.go | 140 + datatransfer/manager.go | 260 ++ datatransfer/message.go | 56 + datatransfer/message/message.go | 19 + datatransfer/message/message1_1/message.go | 195 + .../message/message1_1/transfer_message.go | 58 + .../message1_1/transfer_message_cbor_gen.go | 187 + .../message/message1_1/transfer_request.go | 165 + .../message1_1/transfer_request_cbor_gen.go | 405 +++ .../message/message1_1/transfer_response.go | 126 + .../message1_1/transfer_response_cbor_gen.go | 274 ++ .../message/message1_1prime/message.go | 206 ++ .../message/message1_1prime/schema.go | 29 + .../message/message1_1prime/schema.ipldsch | 37 + .../message1_1prime/transfer_message.go | 43 + .../message1_1prime/transfer_request.go | 146 + .../message1_1prime/transfer_response.go | 115 + datatransfer/message/types/message_types.go | 16 + datatransfer/network/interface.go | 57 + datatransfer/network/libp2p_impl.go | 355 ++ datatransfer/registry/registry.go | 81 + datatransfer/testutil/testutils.go | 24 + datatransfer/tracing/tracing.go | 64 + .../graphsync/extension/gsextension.go | 83 + datatransfer/transport/graphsync/graphsync.go | 1300 +++++++ datatransfer/types.go | 428 +++ datatransfer/types_cbor_gen.go | 487 +++ documentation/en/api-v1-methods.md | 884 +---- fundmanager/fundmanager.go | 10 +- go.mod | 44 +- go.sum | 43 +- go.work.sum | 47 +- gql/module.go | 68 + gql/resolver.go | 16 +- gql/resolver_ask.go | 16 +- gql/resolver_dealpublish.go | 4 +- gql/resolver_legacy.go | 30 +- gql/resolver_legacy_storage.go | 35 - gql/resolver_rtvllog.go | 3 +- gql/resolver_transfers.go | 2 +- gql/schema.graphql | 9 - indexprovider/mock/mock.go | 273 -- indexprovider/wrapper.go | 121 +- indexprovider/wrapper_test.go | 24 +- itests/data_segment_index_retrieval_test.go | 29 +- itests/disabled_markets_v1_deal_test.go | 53 - itests/dummydeal_offline_test.go | 10 +- itests/dummydeal_podsi_test.go | 4 +- itests/dummydeal_test.go | 9 +- itests/framework/fixtures.go | 164 - itests/framework/framework.go | 323 +- itests/framework/log.go | 3 + ...test.go => graphsync_identity_cid_test.go} | 67 +- ...al_test.go => graphsync_retrieval_test.go} | 158 +- itests/ipni_publish_test.go | 1 - itests/markets_v1_deal_test.go | 55 - itests/markets_v1_offline_deal_test.go | 89 - itests/multiminer_retrieval_graphsync_test.go | 8 +- itests/shared/multiminer.go | 8 +- lib/legacy/dealmanager.go | 116 +- lib/legacy/mocks/legacy_manager_mock.go | 154 + markets/journal.go | 76 - markets/loggers/loggers.go | 37 +- markets/piecestore/impl/piecestore.go | 214 ++ markets/piecestore/migrations/migrations.go | 90 + .../migrations/migrations_cbor_gen.go | 527 +++ markets/piecestore/types.go | 70 + markets/piecestore/types_cbor_gen.go | 737 ++++ markets/pricing/cli.go | 50 - markets/retrievaladapter/client.go | 127 - markets/retrievaladapter/client_blockstore.go | 83 - markets/retrievaladapter/provider.go | 108 - markets/retrievaladapter/provider_test.go | 206 -- markets/sectoraccessor/sectoraccessor.go | 4 +- markets/shared/ready.go | 104 + markets/shared/retrystream.go | 103 + markets/shared/shared.go | 7 + markets/shared/timecounter.go | 21 + markets/storageadapter/api.go | 49 - markets/storageadapter/client.go | 445 --- markets/storageadapter/client_blockstore.go | 102 - markets/storageadapter/dealpublisher.go | 13 +- markets/storageadapter/dealpublisher_test.go | 14 + markets/storageadapter/dealstatematcher.go | 85 - .../storageadapter/dealstatematcher_test.go | 155 - .../storageadapter/ondealsectorcommitted.go | 399 --- .../ondealsectorcommitted_test.go | 581 --- markets/storageadapter/provider.go | 438 --- markets/utils/converters.go | 12 +- markets/utils/selectors.go | 45 +- node/builder.go | 160 +- node/config/def.go | 139 +- node/config/doc_gen.go | 385 +- node/config/migrate.go | 3 +- node/config/types.go | 236 +- node/config/v5_to_v6.go | 28 + node/impl/boost.go | 410 +-- node/impl/boost_legacy.go | 239 -- node/modules/client.go | 176 - node/modules/dealfilter.go | 4 +- node/modules/directdeals.go | 6 +- node/modules/dtypes/miner.go | 7 +- node/modules/dtypes/storage.go | 15 +- node/modules/graphsync.go | 71 +- node/modules/legacy_markets.go | 127 - node/modules/piecedirectory.go | 173 +- node/modules/provider_piece_store.go | 25 - node/modules/retrieval.go | 32 +- node/modules/storageminer.go | 440 +-- node/modules/storageminer_dagstore.go | 138 - node/modules/storageminer_idxprov.go | 184 +- react/src/StorageSpace.js | 58 +- react/src/gql.js | 11 - retrievalmarket/client/client.go | 161 +- retrievalmarket/lib/idxciddagstore.go | 72 - retrievalmarket/lib/shardselector.go | 181 - retrievalmarket/lib/shardselector_test.go | 129 - retrievalmarket/lp2pimpl/transports.go | 2 +- retrievalmarket/mock/gen.go | 4 - retrievalmarket/mock/piecestore.go | 152 - retrievalmarket/mock/retrievalmarket.go | 229 -- retrievalmarket/rtvllog/db.go | 28 +- retrievalmarket/rtvllog/retrieval_log.go | 99 +- retrievalmarket/server/channelstate.go | 46 +- .../server/datatransfer.go | 10 +- retrievalmarket/server/events.go | 42 +- retrievalmarket/server/gsunpaidretrieval.go | 179 +- .../server/gsunpaidretrieval_test.go | 384 -- retrievalmarket/server/provider_pieces.go | 6 +- retrievalmarket/server/queryask.go | 34 +- retrievalmarket/server/types.go | 10 +- retrievalmarket/server/validation.go | 48 +- retrievalmarket/testutil/testutil.go | 99 + .../types/legacyretrievaltypes/bindoptions.go | 164 + .../types/legacyretrievaltypes/dealstatus.go | 186 + .../types/legacyretrievaltypes/events.go | 281 ++ .../migrations/maptypes/maptypes.go | 55 + .../migrations/maptypes/maptypes_cbor_gen.go | 1142 ++++++ .../migrations/migrations.go | 386 ++ .../migrations/migrations_cbor_gen.go | 1815 ++++++++++ .../types/legacyretrievaltypes/types.go | 536 +++ .../legacyretrievaltypes/types_cbor_gen.go | 2909 +++++++++++++++ retrievalmarket/types/types.go | 14 + retrievalmarket/types/voucher_legs.go | 38 +- storagemarket/dealfilter/cli.go | 8 +- storagemarket/helper.go | 6 +- storagemarket/lp2pimpl/net.go | 165 +- storagemarket/provider.go | 17 +- storagemarket/provider_test.go | 59 +- storagemarket/smtestutil/mocks.go | 14 +- storagemarket/storedask/create_ask_db.sql | 10 + storagemarket/storedask/db.go | 105 + storagemarket/storedask/storedask.go | 226 ++ .../contract_deal_proposal_types_cbor_gen.go | 2 +- storagemarket/types/legacytypes/dealstatus.go | 235 ++ .../types/legacytypes/filestore/file.go | 39 + .../types/legacytypes/filestore/filestore.go | 82 + .../types/legacytypes/filestore/types.go | 38 + .../legacytypes/migrations/migrations.go | 325 ++ .../migrations/migrations_cbor_gen.go | 2271 ++++++++++++ .../migrations/migrations_mapenc_types.go | 55 + .../migrations_mapenc_types_cbor_gen.go | 936 +++++ .../types/legacytypes/network/types.go | 79 + .../legacytypes/network/types_cbor_gen.go | 927 +++++ storagemarket/types/legacytypes/types.go | 319 ++ .../types/legacytypes/types_cbor_gen.go | 3180 +++++++++++++++++ storagemarket/types/mock_types/mocks.go | 5 +- storagemarket/types/types.go | 13 +- 214 files changed, 32333 insertions(+), 11501 deletions(-) delete mode 100644 cmd/boostd/dagstore.go delete mode 100644 cmd/boostd/legacy_data_transfers.go delete mode 100644 cmd/boostd/legacy_retrieval_deals.go create mode 100644 cmd/lib/stores/dagstore.go create mode 100644 cmd/lib/stores/error.go create mode 100644 cmd/lib/stores/filestore.go create mode 100644 cmd/lib/stores/kvcarbs.go create mode 100644 cmd/lib/stores/ro_bstores.go create mode 100644 cmd/lib/stores/rw_bstores.go create mode 100644 datatransfer/channelmonitor/channelmonitor.go create mode 100644 datatransfer/channels/block_index_cache.go create mode 100644 datatransfer/channels/channel_state.go create mode 100644 datatransfer/channels/channels.go create mode 100644 datatransfer/channels/channels_fsm.go create mode 100644 datatransfer/channels/internal/internalchannel.go create mode 100644 datatransfer/channels/internal/internalchannel_cbor_gen.go create mode 100644 datatransfer/channels/internal/migrations/migrations.go create mode 100644 datatransfer/encoding/encoding.go create mode 100644 datatransfer/errors.go create mode 100644 datatransfer/events.go create mode 100644 datatransfer/impl/environment.go create mode 100644 datatransfer/impl/events.go create mode 100644 datatransfer/impl/impl.go create mode 100644 datatransfer/impl/receiver.go create mode 100644 datatransfer/impl/restart.go create mode 100644 datatransfer/impl/timecounter.go create mode 100644 datatransfer/impl/utils.go create mode 100644 datatransfer/manager.go create mode 100644 datatransfer/message.go create mode 100644 datatransfer/message/message.go create mode 100644 datatransfer/message/message1_1/message.go create mode 100644 datatransfer/message/message1_1/transfer_message.go create mode 100644 datatransfer/message/message1_1/transfer_message_cbor_gen.go create mode 100644 datatransfer/message/message1_1/transfer_request.go create mode 100644 datatransfer/message/message1_1/transfer_request_cbor_gen.go create mode 100644 datatransfer/message/message1_1/transfer_response.go create mode 100644 datatransfer/message/message1_1/transfer_response_cbor_gen.go create mode 100644 datatransfer/message/message1_1prime/message.go create mode 100644 datatransfer/message/message1_1prime/schema.go create mode 100644 datatransfer/message/message1_1prime/schema.ipldsch create mode 100644 datatransfer/message/message1_1prime/transfer_message.go create mode 100644 datatransfer/message/message1_1prime/transfer_request.go create mode 100644 datatransfer/message/message1_1prime/transfer_response.go create mode 100644 datatransfer/message/types/message_types.go create mode 100644 datatransfer/network/interface.go create mode 100644 datatransfer/network/libp2p_impl.go create mode 100644 datatransfer/registry/registry.go create mode 100644 datatransfer/testutil/testutils.go create mode 100644 datatransfer/tracing/tracing.go create mode 100644 datatransfer/transport/graphsync/extension/gsextension.go create mode 100644 datatransfer/transport/graphsync/graphsync.go create mode 100644 datatransfer/types.go create mode 100644 datatransfer/types_cbor_gen.go create mode 100644 gql/module.go delete mode 100644 gql/resolver_legacy_storage.go delete mode 100644 indexprovider/mock/mock.go delete mode 100644 itests/disabled_markets_v1_deal_test.go delete mode 100644 itests/framework/fixtures.go rename itests/{markets_v1_identity_cid_test.go => graphsync_identity_cid_test.go} (68%) rename itests/{markets_v1_retrieval_test.go => graphsync_retrieval_test.go} (61%) delete mode 100644 itests/markets_v1_deal_test.go delete mode 100644 itests/markets_v1_offline_deal_test.go create mode 100644 lib/legacy/mocks/legacy_manager_mock.go delete mode 100644 markets/journal.go create mode 100644 markets/piecestore/impl/piecestore.go create mode 100644 markets/piecestore/migrations/migrations.go create mode 100644 markets/piecestore/migrations/migrations_cbor_gen.go create mode 100644 markets/piecestore/types.go create mode 100644 markets/piecestore/types_cbor_gen.go delete mode 100644 markets/pricing/cli.go delete mode 100644 markets/retrievaladapter/client.go delete mode 100644 markets/retrievaladapter/client_blockstore.go delete mode 100644 markets/retrievaladapter/provider.go delete mode 100644 markets/retrievaladapter/provider_test.go create mode 100644 markets/shared/ready.go create mode 100644 markets/shared/retrystream.go create mode 100644 markets/shared/shared.go create mode 100644 markets/shared/timecounter.go delete mode 100644 markets/storageadapter/api.go delete mode 100644 markets/storageadapter/client.go delete mode 100644 markets/storageadapter/client_blockstore.go delete mode 100644 markets/storageadapter/dealstatematcher.go delete mode 100644 markets/storageadapter/dealstatematcher_test.go delete mode 100644 markets/storageadapter/ondealsectorcommitted.go delete mode 100644 markets/storageadapter/ondealsectorcommitted_test.go delete mode 100644 markets/storageadapter/provider.go create mode 100644 node/config/v5_to_v6.go delete mode 100644 node/impl/boost_legacy.go delete mode 100644 node/modules/client.go delete mode 100644 node/modules/legacy_markets.go delete mode 100644 node/modules/provider_piece_store.go delete mode 100644 node/modules/storageminer_dagstore.go delete mode 100644 retrievalmarket/lib/idxciddagstore.go delete mode 100644 retrievalmarket/lib/shardselector.go delete mode 100644 retrievalmarket/lib/shardselector_test.go delete mode 100644 retrievalmarket/mock/gen.go delete mode 100644 retrievalmarket/mock/piecestore.go delete mode 100644 retrievalmarket/mock/retrievalmarket.go rename node/modules/provider_data_transfer.go => retrievalmarket/server/datatransfer.go (81%) delete mode 100644 retrievalmarket/server/gsunpaidretrieval_test.go create mode 100644 retrievalmarket/testutil/testutil.go create mode 100644 retrievalmarket/types/legacyretrievaltypes/bindoptions.go create mode 100644 retrievalmarket/types/legacyretrievaltypes/dealstatus.go create mode 100644 retrievalmarket/types/legacyretrievaltypes/events.go create mode 100644 retrievalmarket/types/legacyretrievaltypes/migrations/maptypes/maptypes.go create mode 100644 retrievalmarket/types/legacyretrievaltypes/migrations/maptypes/maptypes_cbor_gen.go create mode 100644 retrievalmarket/types/legacyretrievaltypes/migrations/migrations.go create mode 100644 retrievalmarket/types/legacyretrievaltypes/migrations/migrations_cbor_gen.go create mode 100644 retrievalmarket/types/legacyretrievaltypes/types.go create mode 100644 retrievalmarket/types/legacyretrievaltypes/types_cbor_gen.go create mode 100644 retrievalmarket/types/types.go create mode 100644 storagemarket/storedask/create_ask_db.sql create mode 100644 storagemarket/storedask/db.go create mode 100644 storagemarket/storedask/storedask.go create mode 100644 storagemarket/types/legacytypes/dealstatus.go create mode 100644 storagemarket/types/legacytypes/filestore/file.go create mode 100644 storagemarket/types/legacytypes/filestore/filestore.go create mode 100644 storagemarket/types/legacytypes/filestore/types.go create mode 100644 storagemarket/types/legacytypes/migrations/migrations.go create mode 100644 storagemarket/types/legacytypes/migrations/migrations_cbor_gen.go create mode 100644 storagemarket/types/legacytypes/migrations/migrations_mapenc_types.go create mode 100644 storagemarket/types/legacytypes/migrations/migrations_mapenc_types_cbor_gen.go create mode 100644 storagemarket/types/legacytypes/network/types.go create mode 100644 storagemarket/types/legacytypes/network/types_cbor_gen.go create mode 100644 storagemarket/types/legacytypes/types.go create mode 100644 storagemarket/types/legacytypes/types_cbor_gen.go diff --git a/.circleci/config.yml b/.circleci/config.yml index f1a14b37e..eb9595558 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -319,29 +319,14 @@ workflows: target: "./itests/dummydeal_test.go" - test: - name: test-itest-markets_v1_deal - suite: itest-markets_v1_deal - target: "./itests/markets_v1_deal_test.go" + name: test-graphsync_identity_cid + suite: itest-graphsync_identity_cid + target: "./itests/graphsync_identity_cid_test.go" - test: - name: test-itest-markets_v1_identity_cid - suite: itest-markets_v1_identity_cid - target: "./itests/markets_v1_identity_cid_test.go" - - - test: - name: test-itest-markets_v1_offline_deal - suite: itest-markets_v1_offline_deal - target: "./itests/markets_v1_offline_deal_test.go" - - - test: - name: test-itest-markets_v1_retrieval - suite: itest-markets_v1_retrieval - target: "./itests/markets_v1_retrieval_test.go" - - - test: - name: test-itest-disabled_markets_v1_deal - suite: itest-disabled_markets_v1_deal - target: "./itests/disabled_markets_v1_deal_test.go" + name: test-itest-retrieval + suite: itest-retrieval + target: "./itests/graphsync_retrieval_test.go" #- test: #name: test-itest-direct_deal diff --git a/api/api.go b/api/api.go index a7e0ff324..3049694be 100644 --- a/api/api.go +++ b/api/api.go @@ -3,17 +3,10 @@ package api import ( "context" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/boost-gfm/storagemarket" smtypes "github.com/filecoin-project/boost/storagemarket/types" - "github.com/filecoin-project/go-address" - datatransfer "github.com/filecoin-project/go-data-transfer" - "github.com/filecoin-project/go-state-types/abi" - lapi "github.com/filecoin-project/lotus/api" - "github.com/filecoin-project/lotus/chain/types" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" "github.com/google/uuid" "github.com/ipfs/go-cid" - "github.com/libp2p/go-libp2p/core/peer" "github.com/multiformats/go-multihash" ) @@ -42,17 +35,9 @@ type Boost interface { BoostDealBySignedProposalCid(ctx context.Context, proposalCid cid.Cid) (*smtypes.ProviderDealState, error) //perm:admin BoostDummyDeal(context.Context, smtypes.DealParams) (*ProviderDealRejectionInfo, error) //perm:admin BoostIndexerAnnounceDealRemoved(ctx context.Context, propCid cid.Cid) (cid.Cid, error) //perm:admin + BoostLegacyDealByProposalCid(ctx context.Context, propCid cid.Cid) (legacytypes.MinerDeal, error) //perm:admin BoostIndexerAnnounceDeal(ctx context.Context, deal *smtypes.ProviderDealState) (cid.Cid, error) //perm:admin - BoostIndexerAnnounceLegacyDeal(ctx context.Context, proposalCid cid.Cid) error //perm:admin - BoostLegacyDealByProposalCid(ctx context.Context, propCid cid.Cid) (storagemarket.MinerDeal, error) //perm:admin - BoostDagstoreRegisterShard(ctx context.Context, key string) error //perm:admin - BoostDagstoreDestroyShard(ctx context.Context, key string) error //perm:admin - BoostDagstoreInitializeShard(ctx context.Context, key string) error //perm:admin - BoostDagstoreInitializeAll(ctx context.Context, params DagstoreInitializeAllParams) (<-chan DagstoreInitializeAllEvent, error) //perm:admin - BoostDagstoreRecoverShard(ctx context.Context, key string) error //perm:admin - BoostDagstoreGC(ctx context.Context) ([]DagstoreShardResult, error) //perm:admin - BoostDagstorePiecesContainingMultihash(ctx context.Context, mh multihash.Multihash) ([]cid.Cid, error) //perm:read - BoostDagstoreListShards(ctx context.Context) ([]DagstoreShardInfo, error) //perm:admin + BoostIndexerAnnounceLegacyDeal(ctx context.Context, proposalCid cid.Cid) (cid.Cid, error) //perm:admin BoostMakeDeal(context.Context, smtypes.DealParams) (*ProviderDealRejectionInfo, error) //perm:write BoostDirectDeal(ctx context.Context, params smtypes.DirectDealParams) (*ProviderDealRejectionInfo, error) //perm:admin @@ -64,75 +49,6 @@ type Boost interface { // MethodGroup: PieceDirectory PdBuildIndexForPieceCid(ctx context.Context, piececid cid.Cid) error //perm:admin - // RuntimeSubsystems returns the subsystems that are enabled - // in this instance. - RuntimeSubsystems(ctx context.Context) (lapi.MinerSubsystems, error) //perm:read - - // MethodGroup: LegacyMarket - MarketListRetrievalDeals(ctx context.Context) ([]retrievalmarket.ProviderDealState, error) //perm:read - MarketSetRetrievalAsk(ctx context.Context, rask *retrievalmarket.Ask) error //perm:admin - MarketGetRetrievalAsk(ctx context.Context) (*retrievalmarket.Ask, error) //perm:read - MarketSetAsk(ctx context.Context, price types.BigInt, verifiedPrice types.BigInt, duration abi.ChainEpoch, minPieceSize abi.PaddedPieceSize, maxPieceSize abi.PaddedPieceSize) error //perm:admin - MarketGetAsk(ctx context.Context) (*storagemarket.SignedStorageAsk, error) //perm:read - MarketListDataTransfers(ctx context.Context) ([]DataTransferChannel, error) //perm:write - MarketDataTransferUpdates(ctx context.Context) (<-chan DataTransferChannel, error) //perm:write - MarketRestartDataTransfer(ctx context.Context, transferID datatransfer.TransferID, otherPeer peer.ID, isInitiator bool) error //perm:write - MarketCancelDataTransfer(ctx context.Context, transferID datatransfer.TransferID, otherPeer peer.ID, isInitiator bool) error //perm:write - MarketImportDealData(ctx context.Context, propcid cid.Cid, path string) error //perm:write - MarketListIncompleteDeals(ctx context.Context) ([]storagemarket.MinerDeal, error) //perm:read - MarketPendingDeals(ctx context.Context) (lapi.PendingDealInfo, error) //perm:write - SectorsRefs(context.Context) (map[string][]lapi.SealedRef, error) //perm:read - - // MethodGroup: Actor - ActorSectorSize(context.Context, address.Address) (abi.SectorSize, error) //perm:read - - // MethodGroup: Deals - DealsConsiderOnlineStorageDeals(context.Context) (bool, error) //perm:admin - DealsSetConsiderOnlineStorageDeals(context.Context, bool) error //perm:admin - DealsConsiderOnlineRetrievalDeals(context.Context) (bool, error) //perm:admin - DealsSetConsiderOnlineRetrievalDeals(context.Context, bool) error //perm:admin - DealsPieceCidBlocklist(context.Context) ([]cid.Cid, error) //perm:admin - DealsSetPieceCidBlocklist(context.Context, []cid.Cid) error //perm:admin - DealsConsiderOfflineStorageDeals(context.Context) (bool, error) //perm:admin - DealsSetConsiderOfflineStorageDeals(context.Context, bool) error //perm:admin - DealsConsiderOfflineRetrievalDeals(context.Context) (bool, error) //perm:admin - DealsSetConsiderOfflineRetrievalDeals(context.Context, bool) error //perm:admin - DealsConsiderVerifiedStorageDeals(context.Context) (bool, error) //perm:admin - DealsSetConsiderVerifiedStorageDeals(context.Context, bool) error //perm:admin - DealsConsiderUnverifiedStorageDeals(context.Context) (bool, error) //perm:admin - DealsSetConsiderUnverifiedStorageDeals(context.Context, bool) error //perm:admin - // MethodGroup: Misc OnlineBackup(context.Context, string) error //perm:admin } - -// DagstoreShardInfo is the serialized form of dagstore.DagstoreShardInfo that -// we expose through JSON-RPC to avoid clients having to depend on the -// dagstore lib. -type DagstoreShardInfo struct { - Key string - State string - Error string -} - -// DagstoreShardResult enumerates results per shard. -type DagstoreShardResult struct { - Key string - Success bool - Error string -} - -type DagstoreInitializeAllParams struct { - MaxConcurrency int - IncludeSealed bool -} - -// DagstoreInitializeAllEvent represents an initialization event. -type DagstoreInitializeAllEvent struct { - Key string - Event string // "start", "end" - Success bool - Error string - Total int - Current int -} diff --git a/api/api_full.go b/api/api_full.go index e306ddd1e..bd51d90a4 100644 --- a/api/api_full.go +++ b/api/api_full.go @@ -5,17 +5,11 @@ import ( "fmt" "time" - "github.com/ipfs/go-cid" - textselector "github.com/ipld/go-ipld-selector-text-lite" - "github.com/filecoin-project/go-address" "github.com/filecoin-project/go-bitfield" - datatransfer "github.com/filecoin-project/go-data-transfer" "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/go-state-types/big" - - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/boost-gfm/storagemarket" + "github.com/ipfs/go-cid" "github.com/filecoin-project/go-state-types/builtin/v8/paych" "github.com/filecoin-project/lotus/chain/actors/builtin" @@ -89,29 +83,6 @@ type Import struct { CARPath string } -type DealInfo struct { - ProposalCid cid.Cid - State storagemarket.StorageDealStatus - Message string // more information about deal state, particularly errors - DealStages *storagemarket.DealStages - Provider address.Address - - DataRef *storagemarket.DataRef - PieceCID cid.Cid - Size uint64 - - PricePerEpoch types.BigInt - Duration uint64 - - DealID abi.DealID - - CreationTime time.Time - Verified bool - - TransferChannelID *datatransfer.ChannelID - DataTransfer *DataTransferChannel -} - type MsgLookup struct { Message cid.Cid // Can be different than requested, in case it was replaced, but only gas values changed Receipt types.MessageReceipt @@ -222,37 +193,6 @@ type MinerPower struct { HasMinPower bool } -type QueryOffer struct { - Err string - - Root cid.Cid - Piece *cid.Cid - - Size uint64 - MinPrice types.BigInt - UnsealPrice types.BigInt - PaymentInterval uint64 - PaymentIntervalIncrease uint64 - Miner address.Address - MinerPeer retrievalmarket.RetrievalPeer -} - -func (o *QueryOffer) Order(client address.Address) RetrievalOrder { - return RetrievalOrder{ - Root: o.Root, - Piece: o.Piece, - Size: o.Size, - Total: o.MinPrice, - UnsealPrice: o.UnsealPrice, - PaymentInterval: o.PaymentInterval, - PaymentIntervalIncrease: o.PaymentIntervalIncrease, - Client: client, - - Miner: o.Miner, - MinerPeer: &o.MinerPeer, - } -} - type MarketBalance struct { Escrow big.Int Locked big.Int @@ -263,24 +203,6 @@ type MarketDeal struct { State market.DealState } -type RetrievalOrder struct { - // TODO: make this less unixfs specific - Root cid.Cid - Piece *cid.Cid - DatamodelPathSelector *textselector.Expression - Size uint64 - - FromLocalCAR string // if specified, get data from a local CARv2 file. - // TODO: support offset - Total types.BigInt - UnsealPrice types.BigInt - PaymentInterval uint64 - PaymentIntervalIncrease uint64 - Client address.Address - Miner address.Address - MinerPeer *retrievalmarket.RetrievalPeer -} - type InvocResult struct { MsgCid cid.Cid Msg *types.Message diff --git a/api/docgen/docgen.go b/api/docgen/docgen.go index ab89bf121..0af0731bb 100644 --- a/api/docgen/docgen.go +++ b/api/docgen/docgen.go @@ -12,11 +12,11 @@ import ( "time" "unicode" - "github.com/filecoin-project/boost-gfm/filestore" - "github.com/filecoin-project/boost-gfm/retrievalmarket" + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes/filestore" "github.com/filecoin-project/go-address" "github.com/filecoin-project/go-bitfield" - datatransfer "github.com/filecoin-project/go-data-transfer" "github.com/filecoin-project/go-jsonrpc/auth" "github.com/google/uuid" "github.com/ipfs/go-cid" @@ -94,7 +94,7 @@ func init() { storeIDExample := imports.ID(50) textSelExample := textselector.Expression("Links/21/Hash/Links/42/Hash") - clientEvent := retrievalmarket.ClientEventDealAccepted + clientEvent := legacyretrievaltypes.ClientEventDealAccepted addExample(bitfield.NewFromSet([]uint64{5})) addExample(abi.RegisteredSealProof_StackedDrg32GiBV1_1) @@ -129,14 +129,15 @@ func init() { addExample(&storeIDExample) addExample(clientEvent) addExample(&clientEvent) - addExample(retrievalmarket.ClientEventDealAccepted) - addExample(retrievalmarket.DealStatusNew) + addExample(legacyretrievaltypes.ClientEventDealAccepted) + addExample(legacyretrievaltypes.DealStatusNew) addExample(&textSelExample) addExample(network.ReachabilityPublic) addExample(build.TestNetworkVersion) allocationID := verifreg.AllocationId(0) addExample(allocationID) addExample(&allocationID) + addExample(filestore.Path("")) addExample(map[string]int{"name": 42}) addExample(map[string]time.Time{"name": time.Unix(1615243938, 0).UTC()}) addExample(&types.ExecutionTrace{ @@ -186,10 +187,9 @@ func init() { ExampleValues[reflect.TypeOf(struct{ A multiaddr.Multiaddr }{}).Field(0).Type] = maddr // miner specific - addExample(filestore.Path(".lotusminer/fstmp123")) si := uint64(12) addExample(&si) - addExample(retrievalmarket.DealID(5)) + addExample(legacyretrievaltypes.DealID(5)) addExample(abi.ActorID(1000)) addExample(storiface.ID("76f1988b-ef30-4d7e-b3ec-9a627f4ba5a8")) addExample(storiface.FTUnsealed) @@ -273,15 +273,6 @@ func init() { addExample(api.CheckStatusCode(0)) addExample(map[string]interface{}{"abc": 123}) - addExample(api.DagstoreShardResult{ - Key: "baga6ea4seaqecmtz7iak33dsfshi627abz4i4665dfuzr3qfs4bmad6dx3iigdq", - Error: "", - }) - addExample(api.DagstoreShardInfo{ - Key: "baga6ea4seaqecmtz7iak33dsfshi627abz4i4665dfuzr3qfs4bmad6dx3iigdq", - State: "ShardStateAvailable", - Error: "", - }) addExample(storiface.ResourceTable) addExample(network.ScopeStat{ Memory: 123, diff --git a/api/proxy_gen.go b/api/proxy_gen.go index c788d8c7e..337d7c174 100644 --- a/api/proxy_gen.go +++ b/api/proxy_gen.go @@ -7,15 +7,11 @@ import ( "errors" "time" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/boost-gfm/storagemarket" smtypes "github.com/filecoin-project/boost/storagemarket/types" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" "github.com/filecoin-project/go-address" - datatransfer "github.com/filecoin-project/go-data-transfer" "github.com/filecoin-project/go-jsonrpc/auth" - "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/go-state-types/crypto" - lapi "github.com/filecoin-project/lotus/api" lotus_api "github.com/filecoin-project/lotus/api" apitypes "github.com/filecoin-project/lotus/api/types" "github.com/filecoin-project/lotus/chain/types" @@ -36,30 +32,12 @@ type BoostStruct struct { NetStruct Internal struct { - ActorSectorSize func(p0 context.Context, p1 address.Address) (abi.SectorSize, error) `perm:"read"` - BlockstoreGet func(p0 context.Context, p1 cid.Cid) ([]byte, error) `perm:"read"` BlockstoreGetSize func(p0 context.Context, p1 cid.Cid) (int, error) `perm:"read"` BlockstoreHas func(p0 context.Context, p1 cid.Cid) (bool, error) `perm:"read"` - BoostDagstoreDestroyShard func(p0 context.Context, p1 string) error `perm:"admin"` - - BoostDagstoreGC func(p0 context.Context) ([]DagstoreShardResult, error) `perm:"admin"` - - BoostDagstoreInitializeAll func(p0 context.Context, p1 DagstoreInitializeAllParams) (<-chan DagstoreInitializeAllEvent, error) `perm:"admin"` - - BoostDagstoreInitializeShard func(p0 context.Context, p1 string) error `perm:"admin"` - - BoostDagstoreListShards func(p0 context.Context) ([]DagstoreShardInfo, error) `perm:"admin"` - - BoostDagstorePiecesContainingMultihash func(p0 context.Context, p1 multihash.Multihash) ([]cid.Cid, error) `perm:"read"` - - BoostDagstoreRecoverShard func(p0 context.Context, p1 string) error `perm:"admin"` - - BoostDagstoreRegisterShard func(p0 context.Context, p1 string) error `perm:"admin"` - BoostDeal func(p0 context.Context, p1 uuid.UUID) (*smtypes.ProviderDealState, error) `perm:"admin"` BoostDealBySignedProposalCid func(p0 context.Context, p1 cid.Cid) (*smtypes.ProviderDealState, error) `perm:"admin"` @@ -78,75 +56,19 @@ type BoostStruct struct { BoostIndexerAnnounceLatestHttp func(p0 context.Context, p1 []string) (cid.Cid, error) `perm:"admin"` - BoostIndexerAnnounceLegacyDeal func(p0 context.Context, p1 cid.Cid) error `perm:"admin"` + BoostIndexerAnnounceLegacyDeal func(p0 context.Context, p1 cid.Cid) (cid.Cid, error) `perm:"admin"` BoostIndexerListMultihashes func(p0 context.Context, p1 cid.Cid) ([]multihash.Multihash, error) `perm:"admin"` - BoostLegacyDealByProposalCid func(p0 context.Context, p1 cid.Cid) (storagemarket.MinerDeal, error) `perm:"admin"` + BoostLegacyDealByProposalCid func(p0 context.Context, p1 cid.Cid) (legacytypes.MinerDeal, error) `perm:"admin"` BoostMakeDeal func(p0 context.Context, p1 smtypes.DealParams) (*ProviderDealRejectionInfo, error) `perm:"write"` BoostOfflineDealWithData func(p0 context.Context, p1 uuid.UUID, p2 string, p3 bool) (*ProviderDealRejectionInfo, error) `perm:"admin"` - DealsConsiderOfflineRetrievalDeals func(p0 context.Context) (bool, error) `perm:"admin"` - - DealsConsiderOfflineStorageDeals func(p0 context.Context) (bool, error) `perm:"admin"` - - DealsConsiderOnlineRetrievalDeals func(p0 context.Context) (bool, error) `perm:"admin"` - - DealsConsiderOnlineStorageDeals func(p0 context.Context) (bool, error) `perm:"admin"` - - DealsConsiderUnverifiedStorageDeals func(p0 context.Context) (bool, error) `perm:"admin"` - - DealsConsiderVerifiedStorageDeals func(p0 context.Context) (bool, error) `perm:"admin"` - - DealsPieceCidBlocklist func(p0 context.Context) ([]cid.Cid, error) `perm:"admin"` - - DealsSetConsiderOfflineRetrievalDeals func(p0 context.Context, p1 bool) error `perm:"admin"` - - DealsSetConsiderOfflineStorageDeals func(p0 context.Context, p1 bool) error `perm:"admin"` - - DealsSetConsiderOnlineRetrievalDeals func(p0 context.Context, p1 bool) error `perm:"admin"` - - DealsSetConsiderOnlineStorageDeals func(p0 context.Context, p1 bool) error `perm:"admin"` - - DealsSetConsiderUnverifiedStorageDeals func(p0 context.Context, p1 bool) error `perm:"admin"` - - DealsSetConsiderVerifiedStorageDeals func(p0 context.Context, p1 bool) error `perm:"admin"` - - DealsSetPieceCidBlocklist func(p0 context.Context, p1 []cid.Cid) error `perm:"admin"` - - MarketCancelDataTransfer func(p0 context.Context, p1 datatransfer.TransferID, p2 peer.ID, p3 bool) error `perm:"write"` - - MarketDataTransferUpdates func(p0 context.Context) (<-chan DataTransferChannel, error) `perm:"write"` - - MarketGetAsk func(p0 context.Context) (*storagemarket.SignedStorageAsk, error) `perm:"read"` - - MarketGetRetrievalAsk func(p0 context.Context) (*retrievalmarket.Ask, error) `perm:"read"` - - MarketImportDealData func(p0 context.Context, p1 cid.Cid, p2 string) error `perm:"write"` - - MarketListDataTransfers func(p0 context.Context) ([]DataTransferChannel, error) `perm:"write"` - - MarketListIncompleteDeals func(p0 context.Context) ([]storagemarket.MinerDeal, error) `perm:"read"` - - MarketListRetrievalDeals func(p0 context.Context) ([]retrievalmarket.ProviderDealState, error) `perm:"read"` - - MarketPendingDeals func(p0 context.Context) (lapi.PendingDealInfo, error) `perm:"write"` - - MarketRestartDataTransfer func(p0 context.Context, p1 datatransfer.TransferID, p2 peer.ID, p3 bool) error `perm:"write"` - - MarketSetAsk func(p0 context.Context, p1 types.BigInt, p2 types.BigInt, p3 abi.ChainEpoch, p4 abi.PaddedPieceSize, p5 abi.PaddedPieceSize) error `perm:"admin"` - - MarketSetRetrievalAsk func(p0 context.Context, p1 *retrievalmarket.Ask) error `perm:"admin"` - OnlineBackup func(p0 context.Context, p1 string) error `perm:"admin"` PdBuildIndexForPieceCid func(p0 context.Context, p1 cid.Cid) error `perm:"admin"` - - RuntimeSubsystems func(p0 context.Context) (lapi.MinerSubsystems, error) `perm:"read"` - - SectorsRefs func(p0 context.Context) (map[string][]lapi.SealedRef, error) `perm:"read"` } } @@ -275,17 +197,6 @@ type WalletStruct struct { type WalletStub struct { } -func (s *BoostStruct) ActorSectorSize(p0 context.Context, p1 address.Address) (abi.SectorSize, error) { - if s.Internal.ActorSectorSize == nil { - return *new(abi.SectorSize), ErrNotSupported - } - return s.Internal.ActorSectorSize(p0, p1) -} - -func (s *BoostStub) ActorSectorSize(p0 context.Context, p1 address.Address) (abi.SectorSize, error) { - return *new(abi.SectorSize), ErrNotSupported -} - func (s *BoostStruct) BlockstoreGet(p0 context.Context, p1 cid.Cid) ([]byte, error) { if s.Internal.BlockstoreGet == nil { return *new([]byte), ErrNotSupported @@ -319,94 +230,6 @@ func (s *BoostStub) BlockstoreHas(p0 context.Context, p1 cid.Cid) (bool, error) return false, ErrNotSupported } -func (s *BoostStruct) BoostDagstoreDestroyShard(p0 context.Context, p1 string) error { - if s.Internal.BoostDagstoreDestroyShard == nil { - return ErrNotSupported - } - return s.Internal.BoostDagstoreDestroyShard(p0, p1) -} - -func (s *BoostStub) BoostDagstoreDestroyShard(p0 context.Context, p1 string) error { - return ErrNotSupported -} - -func (s *BoostStruct) BoostDagstoreGC(p0 context.Context) ([]DagstoreShardResult, error) { - if s.Internal.BoostDagstoreGC == nil { - return *new([]DagstoreShardResult), ErrNotSupported - } - return s.Internal.BoostDagstoreGC(p0) -} - -func (s *BoostStub) BoostDagstoreGC(p0 context.Context) ([]DagstoreShardResult, error) { - return *new([]DagstoreShardResult), ErrNotSupported -} - -func (s *BoostStruct) BoostDagstoreInitializeAll(p0 context.Context, p1 DagstoreInitializeAllParams) (<-chan DagstoreInitializeAllEvent, error) { - if s.Internal.BoostDagstoreInitializeAll == nil { - return nil, ErrNotSupported - } - return s.Internal.BoostDagstoreInitializeAll(p0, p1) -} - -func (s *BoostStub) BoostDagstoreInitializeAll(p0 context.Context, p1 DagstoreInitializeAllParams) (<-chan DagstoreInitializeAllEvent, error) { - return nil, ErrNotSupported -} - -func (s *BoostStruct) BoostDagstoreInitializeShard(p0 context.Context, p1 string) error { - if s.Internal.BoostDagstoreInitializeShard == nil { - return ErrNotSupported - } - return s.Internal.BoostDagstoreInitializeShard(p0, p1) -} - -func (s *BoostStub) BoostDagstoreInitializeShard(p0 context.Context, p1 string) error { - return ErrNotSupported -} - -func (s *BoostStruct) BoostDagstoreListShards(p0 context.Context) ([]DagstoreShardInfo, error) { - if s.Internal.BoostDagstoreListShards == nil { - return *new([]DagstoreShardInfo), ErrNotSupported - } - return s.Internal.BoostDagstoreListShards(p0) -} - -func (s *BoostStub) BoostDagstoreListShards(p0 context.Context) ([]DagstoreShardInfo, error) { - return *new([]DagstoreShardInfo), ErrNotSupported -} - -func (s *BoostStruct) BoostDagstorePiecesContainingMultihash(p0 context.Context, p1 multihash.Multihash) ([]cid.Cid, error) { - if s.Internal.BoostDagstorePiecesContainingMultihash == nil { - return *new([]cid.Cid), ErrNotSupported - } - return s.Internal.BoostDagstorePiecesContainingMultihash(p0, p1) -} - -func (s *BoostStub) BoostDagstorePiecesContainingMultihash(p0 context.Context, p1 multihash.Multihash) ([]cid.Cid, error) { - return *new([]cid.Cid), ErrNotSupported -} - -func (s *BoostStruct) BoostDagstoreRecoverShard(p0 context.Context, p1 string) error { - if s.Internal.BoostDagstoreRecoverShard == nil { - return ErrNotSupported - } - return s.Internal.BoostDagstoreRecoverShard(p0, p1) -} - -func (s *BoostStub) BoostDagstoreRecoverShard(p0 context.Context, p1 string) error { - return ErrNotSupported -} - -func (s *BoostStruct) BoostDagstoreRegisterShard(p0 context.Context, p1 string) error { - if s.Internal.BoostDagstoreRegisterShard == nil { - return ErrNotSupported - } - return s.Internal.BoostDagstoreRegisterShard(p0, p1) -} - -func (s *BoostStub) BoostDagstoreRegisterShard(p0 context.Context, p1 string) error { - return ErrNotSupported -} - func (s *BoostStruct) BoostDeal(p0 context.Context, p1 uuid.UUID) (*smtypes.ProviderDealState, error) { if s.Internal.BoostDeal == nil { return nil, ErrNotSupported @@ -506,15 +329,15 @@ func (s *BoostStub) BoostIndexerAnnounceLatestHttp(p0 context.Context, p1 []stri return *new(cid.Cid), ErrNotSupported } -func (s *BoostStruct) BoostIndexerAnnounceLegacyDeal(p0 context.Context, p1 cid.Cid) error { +func (s *BoostStruct) BoostIndexerAnnounceLegacyDeal(p0 context.Context, p1 cid.Cid) (cid.Cid, error) { if s.Internal.BoostIndexerAnnounceLegacyDeal == nil { - return ErrNotSupported + return *new(cid.Cid), ErrNotSupported } return s.Internal.BoostIndexerAnnounceLegacyDeal(p0, p1) } -func (s *BoostStub) BoostIndexerAnnounceLegacyDeal(p0 context.Context, p1 cid.Cid) error { - return ErrNotSupported +func (s *BoostStub) BoostIndexerAnnounceLegacyDeal(p0 context.Context, p1 cid.Cid) (cid.Cid, error) { + return *new(cid.Cid), ErrNotSupported } func (s *BoostStruct) BoostIndexerListMultihashes(p0 context.Context, p1 cid.Cid) ([]multihash.Multihash, error) { @@ -528,15 +351,15 @@ func (s *BoostStub) BoostIndexerListMultihashes(p0 context.Context, p1 cid.Cid) return *new([]multihash.Multihash), ErrNotSupported } -func (s *BoostStruct) BoostLegacyDealByProposalCid(p0 context.Context, p1 cid.Cid) (storagemarket.MinerDeal, error) { +func (s *BoostStruct) BoostLegacyDealByProposalCid(p0 context.Context, p1 cid.Cid) (legacytypes.MinerDeal, error) { if s.Internal.BoostLegacyDealByProposalCid == nil { - return *new(storagemarket.MinerDeal), ErrNotSupported + return *new(legacytypes.MinerDeal), ErrNotSupported } return s.Internal.BoostLegacyDealByProposalCid(p0, p1) } -func (s *BoostStub) BoostLegacyDealByProposalCid(p0 context.Context, p1 cid.Cid) (storagemarket.MinerDeal, error) { - return *new(storagemarket.MinerDeal), ErrNotSupported +func (s *BoostStub) BoostLegacyDealByProposalCid(p0 context.Context, p1 cid.Cid) (legacytypes.MinerDeal, error) { + return *new(legacytypes.MinerDeal), ErrNotSupported } func (s *BoostStruct) BoostMakeDeal(p0 context.Context, p1 smtypes.DealParams) (*ProviderDealRejectionInfo, error) { @@ -561,292 +384,6 @@ func (s *BoostStub) BoostOfflineDealWithData(p0 context.Context, p1 uuid.UUID, p return nil, ErrNotSupported } -func (s *BoostStruct) DealsConsiderOfflineRetrievalDeals(p0 context.Context) (bool, error) { - if s.Internal.DealsConsiderOfflineRetrievalDeals == nil { - return false, ErrNotSupported - } - return s.Internal.DealsConsiderOfflineRetrievalDeals(p0) -} - -func (s *BoostStub) DealsConsiderOfflineRetrievalDeals(p0 context.Context) (bool, error) { - return false, ErrNotSupported -} - -func (s *BoostStruct) DealsConsiderOfflineStorageDeals(p0 context.Context) (bool, error) { - if s.Internal.DealsConsiderOfflineStorageDeals == nil { - return false, ErrNotSupported - } - return s.Internal.DealsConsiderOfflineStorageDeals(p0) -} - -func (s *BoostStub) DealsConsiderOfflineStorageDeals(p0 context.Context) (bool, error) { - return false, ErrNotSupported -} - -func (s *BoostStruct) DealsConsiderOnlineRetrievalDeals(p0 context.Context) (bool, error) { - if s.Internal.DealsConsiderOnlineRetrievalDeals == nil { - return false, ErrNotSupported - } - return s.Internal.DealsConsiderOnlineRetrievalDeals(p0) -} - -func (s *BoostStub) DealsConsiderOnlineRetrievalDeals(p0 context.Context) (bool, error) { - return false, ErrNotSupported -} - -func (s *BoostStruct) DealsConsiderOnlineStorageDeals(p0 context.Context) (bool, error) { - if s.Internal.DealsConsiderOnlineStorageDeals == nil { - return false, ErrNotSupported - } - return s.Internal.DealsConsiderOnlineStorageDeals(p0) -} - -func (s *BoostStub) DealsConsiderOnlineStorageDeals(p0 context.Context) (bool, error) { - return false, ErrNotSupported -} - -func (s *BoostStruct) DealsConsiderUnverifiedStorageDeals(p0 context.Context) (bool, error) { - if s.Internal.DealsConsiderUnverifiedStorageDeals == nil { - return false, ErrNotSupported - } - return s.Internal.DealsConsiderUnverifiedStorageDeals(p0) -} - -func (s *BoostStub) DealsConsiderUnverifiedStorageDeals(p0 context.Context) (bool, error) { - return false, ErrNotSupported -} - -func (s *BoostStruct) DealsConsiderVerifiedStorageDeals(p0 context.Context) (bool, error) { - if s.Internal.DealsConsiderVerifiedStorageDeals == nil { - return false, ErrNotSupported - } - return s.Internal.DealsConsiderVerifiedStorageDeals(p0) -} - -func (s *BoostStub) DealsConsiderVerifiedStorageDeals(p0 context.Context) (bool, error) { - return false, ErrNotSupported -} - -func (s *BoostStruct) DealsPieceCidBlocklist(p0 context.Context) ([]cid.Cid, error) { - if s.Internal.DealsPieceCidBlocklist == nil { - return *new([]cid.Cid), ErrNotSupported - } - return s.Internal.DealsPieceCidBlocklist(p0) -} - -func (s *BoostStub) DealsPieceCidBlocklist(p0 context.Context) ([]cid.Cid, error) { - return *new([]cid.Cid), ErrNotSupported -} - -func (s *BoostStruct) DealsSetConsiderOfflineRetrievalDeals(p0 context.Context, p1 bool) error { - if s.Internal.DealsSetConsiderOfflineRetrievalDeals == nil { - return ErrNotSupported - } - return s.Internal.DealsSetConsiderOfflineRetrievalDeals(p0, p1) -} - -func (s *BoostStub) DealsSetConsiderOfflineRetrievalDeals(p0 context.Context, p1 bool) error { - return ErrNotSupported -} - -func (s *BoostStruct) DealsSetConsiderOfflineStorageDeals(p0 context.Context, p1 bool) error { - if s.Internal.DealsSetConsiderOfflineStorageDeals == nil { - return ErrNotSupported - } - return s.Internal.DealsSetConsiderOfflineStorageDeals(p0, p1) -} - -func (s *BoostStub) DealsSetConsiderOfflineStorageDeals(p0 context.Context, p1 bool) error { - return ErrNotSupported -} - -func (s *BoostStruct) DealsSetConsiderOnlineRetrievalDeals(p0 context.Context, p1 bool) error { - if s.Internal.DealsSetConsiderOnlineRetrievalDeals == nil { - return ErrNotSupported - } - return s.Internal.DealsSetConsiderOnlineRetrievalDeals(p0, p1) -} - -func (s *BoostStub) DealsSetConsiderOnlineRetrievalDeals(p0 context.Context, p1 bool) error { - return ErrNotSupported -} - -func (s *BoostStruct) DealsSetConsiderOnlineStorageDeals(p0 context.Context, p1 bool) error { - if s.Internal.DealsSetConsiderOnlineStorageDeals == nil { - return ErrNotSupported - } - return s.Internal.DealsSetConsiderOnlineStorageDeals(p0, p1) -} - -func (s *BoostStub) DealsSetConsiderOnlineStorageDeals(p0 context.Context, p1 bool) error { - return ErrNotSupported -} - -func (s *BoostStruct) DealsSetConsiderUnverifiedStorageDeals(p0 context.Context, p1 bool) error { - if s.Internal.DealsSetConsiderUnverifiedStorageDeals == nil { - return ErrNotSupported - } - return s.Internal.DealsSetConsiderUnverifiedStorageDeals(p0, p1) -} - -func (s *BoostStub) DealsSetConsiderUnverifiedStorageDeals(p0 context.Context, p1 bool) error { - return ErrNotSupported -} - -func (s *BoostStruct) DealsSetConsiderVerifiedStorageDeals(p0 context.Context, p1 bool) error { - if s.Internal.DealsSetConsiderVerifiedStorageDeals == nil { - return ErrNotSupported - } - return s.Internal.DealsSetConsiderVerifiedStorageDeals(p0, p1) -} - -func (s *BoostStub) DealsSetConsiderVerifiedStorageDeals(p0 context.Context, p1 bool) error { - return ErrNotSupported -} - -func (s *BoostStruct) DealsSetPieceCidBlocklist(p0 context.Context, p1 []cid.Cid) error { - if s.Internal.DealsSetPieceCidBlocklist == nil { - return ErrNotSupported - } - return s.Internal.DealsSetPieceCidBlocklist(p0, p1) -} - -func (s *BoostStub) DealsSetPieceCidBlocklist(p0 context.Context, p1 []cid.Cid) error { - return ErrNotSupported -} - -func (s *BoostStruct) MarketCancelDataTransfer(p0 context.Context, p1 datatransfer.TransferID, p2 peer.ID, p3 bool) error { - if s.Internal.MarketCancelDataTransfer == nil { - return ErrNotSupported - } - return s.Internal.MarketCancelDataTransfer(p0, p1, p2, p3) -} - -func (s *BoostStub) MarketCancelDataTransfer(p0 context.Context, p1 datatransfer.TransferID, p2 peer.ID, p3 bool) error { - return ErrNotSupported -} - -func (s *BoostStruct) MarketDataTransferUpdates(p0 context.Context) (<-chan DataTransferChannel, error) { - if s.Internal.MarketDataTransferUpdates == nil { - return nil, ErrNotSupported - } - return s.Internal.MarketDataTransferUpdates(p0) -} - -func (s *BoostStub) MarketDataTransferUpdates(p0 context.Context) (<-chan DataTransferChannel, error) { - return nil, ErrNotSupported -} - -func (s *BoostStruct) MarketGetAsk(p0 context.Context) (*storagemarket.SignedStorageAsk, error) { - if s.Internal.MarketGetAsk == nil { - return nil, ErrNotSupported - } - return s.Internal.MarketGetAsk(p0) -} - -func (s *BoostStub) MarketGetAsk(p0 context.Context) (*storagemarket.SignedStorageAsk, error) { - return nil, ErrNotSupported -} - -func (s *BoostStruct) MarketGetRetrievalAsk(p0 context.Context) (*retrievalmarket.Ask, error) { - if s.Internal.MarketGetRetrievalAsk == nil { - return nil, ErrNotSupported - } - return s.Internal.MarketGetRetrievalAsk(p0) -} - -func (s *BoostStub) MarketGetRetrievalAsk(p0 context.Context) (*retrievalmarket.Ask, error) { - return nil, ErrNotSupported -} - -func (s *BoostStruct) MarketImportDealData(p0 context.Context, p1 cid.Cid, p2 string) error { - if s.Internal.MarketImportDealData == nil { - return ErrNotSupported - } - return s.Internal.MarketImportDealData(p0, p1, p2) -} - -func (s *BoostStub) MarketImportDealData(p0 context.Context, p1 cid.Cid, p2 string) error { - return ErrNotSupported -} - -func (s *BoostStruct) MarketListDataTransfers(p0 context.Context) ([]DataTransferChannel, error) { - if s.Internal.MarketListDataTransfers == nil { - return *new([]DataTransferChannel), ErrNotSupported - } - return s.Internal.MarketListDataTransfers(p0) -} - -func (s *BoostStub) MarketListDataTransfers(p0 context.Context) ([]DataTransferChannel, error) { - return *new([]DataTransferChannel), ErrNotSupported -} - -func (s *BoostStruct) MarketListIncompleteDeals(p0 context.Context) ([]storagemarket.MinerDeal, error) { - if s.Internal.MarketListIncompleteDeals == nil { - return *new([]storagemarket.MinerDeal), ErrNotSupported - } - return s.Internal.MarketListIncompleteDeals(p0) -} - -func (s *BoostStub) MarketListIncompleteDeals(p0 context.Context) ([]storagemarket.MinerDeal, error) { - return *new([]storagemarket.MinerDeal), ErrNotSupported -} - -func (s *BoostStruct) MarketListRetrievalDeals(p0 context.Context) ([]retrievalmarket.ProviderDealState, error) { - if s.Internal.MarketListRetrievalDeals == nil { - return *new([]retrievalmarket.ProviderDealState), ErrNotSupported - } - return s.Internal.MarketListRetrievalDeals(p0) -} - -func (s *BoostStub) MarketListRetrievalDeals(p0 context.Context) ([]retrievalmarket.ProviderDealState, error) { - return *new([]retrievalmarket.ProviderDealState), ErrNotSupported -} - -func (s *BoostStruct) MarketPendingDeals(p0 context.Context) (lapi.PendingDealInfo, error) { - if s.Internal.MarketPendingDeals == nil { - return *new(lapi.PendingDealInfo), ErrNotSupported - } - return s.Internal.MarketPendingDeals(p0) -} - -func (s *BoostStub) MarketPendingDeals(p0 context.Context) (lapi.PendingDealInfo, error) { - return *new(lapi.PendingDealInfo), ErrNotSupported -} - -func (s *BoostStruct) MarketRestartDataTransfer(p0 context.Context, p1 datatransfer.TransferID, p2 peer.ID, p3 bool) error { - if s.Internal.MarketRestartDataTransfer == nil { - return ErrNotSupported - } - return s.Internal.MarketRestartDataTransfer(p0, p1, p2, p3) -} - -func (s *BoostStub) MarketRestartDataTransfer(p0 context.Context, p1 datatransfer.TransferID, p2 peer.ID, p3 bool) error { - return ErrNotSupported -} - -func (s *BoostStruct) MarketSetAsk(p0 context.Context, p1 types.BigInt, p2 types.BigInt, p3 abi.ChainEpoch, p4 abi.PaddedPieceSize, p5 abi.PaddedPieceSize) error { - if s.Internal.MarketSetAsk == nil { - return ErrNotSupported - } - return s.Internal.MarketSetAsk(p0, p1, p2, p3, p4, p5) -} - -func (s *BoostStub) MarketSetAsk(p0 context.Context, p1 types.BigInt, p2 types.BigInt, p3 abi.ChainEpoch, p4 abi.PaddedPieceSize, p5 abi.PaddedPieceSize) error { - return ErrNotSupported -} - -func (s *BoostStruct) MarketSetRetrievalAsk(p0 context.Context, p1 *retrievalmarket.Ask) error { - if s.Internal.MarketSetRetrievalAsk == nil { - return ErrNotSupported - } - return s.Internal.MarketSetRetrievalAsk(p0, p1) -} - -func (s *BoostStub) MarketSetRetrievalAsk(p0 context.Context, p1 *retrievalmarket.Ask) error { - return ErrNotSupported -} - func (s *BoostStruct) OnlineBackup(p0 context.Context, p1 string) error { if s.Internal.OnlineBackup == nil { return ErrNotSupported @@ -869,28 +406,6 @@ func (s *BoostStub) PdBuildIndexForPieceCid(p0 context.Context, p1 cid.Cid) erro return ErrNotSupported } -func (s *BoostStruct) RuntimeSubsystems(p0 context.Context) (lapi.MinerSubsystems, error) { - if s.Internal.RuntimeSubsystems == nil { - return *new(lapi.MinerSubsystems), ErrNotSupported - } - return s.Internal.RuntimeSubsystems(p0) -} - -func (s *BoostStub) RuntimeSubsystems(p0 context.Context) (lapi.MinerSubsystems, error) { - return *new(lapi.MinerSubsystems), ErrNotSupported -} - -func (s *BoostStruct) SectorsRefs(p0 context.Context) (map[string][]lapi.SealedRef, error) { - if s.Internal.SectorsRefs == nil { - return *new(map[string][]lapi.SealedRef), ErrNotSupported - } - return s.Internal.SectorsRefs(p0) -} - -func (s *BoostStub) SectorsRefs(p0 context.Context) (map[string][]lapi.SealedRef, error) { - return *new(map[string][]lapi.SealedRef), ErrNotSupported -} - func (s *ChainIOStruct) ChainHasObj(p0 context.Context, p1 cid.Cid) (bool, error) { if s.Internal.ChainHasObj == nil { return false, ErrNotSupported diff --git a/api/types.go b/api/types.go index a80fe952b..93558f324 100644 --- a/api/types.go +++ b/api/types.go @@ -2,13 +2,10 @@ package api import ( "encoding/json" - "fmt" "time" - "github.com/filecoin-project/boost-gfm/retrievalmarket" "github.com/filecoin-project/lotus/chain/types" - datatransfer "github.com/filecoin-project/go-data-transfer" "github.com/filecoin-project/go-state-types/abi" "github.com/ipfs/go-cid" @@ -61,51 +58,6 @@ type MessageSendSpec struct { MaxFee abi.TokenAmount } -type DataTransferChannel struct { - TransferID datatransfer.TransferID - Status datatransfer.Status - BaseCID cid.Cid - IsInitiator bool - IsSender bool - Voucher string - Message string - OtherPeer peer.ID - Transferred uint64 - Stages *datatransfer.ChannelStages -} - -// NewDataTransferChannel constructs an API DataTransferChannel type from full channel state snapshot and a host id -func NewDataTransferChannel(hostID peer.ID, channelState datatransfer.ChannelState) DataTransferChannel { - channel := DataTransferChannel{ - TransferID: channelState.TransferID(), - Status: channelState.Status(), - BaseCID: channelState.BaseCID(), - IsSender: channelState.Sender() == hostID, - Message: channelState.Message(), - } - stringer, ok := channelState.Voucher().(fmt.Stringer) - if ok { - channel.Voucher = stringer.String() - } else { - voucherJSON, err := json.Marshal(channelState.Voucher()) - if err != nil { - channel.Voucher = fmt.Sprintf("Voucher Serialization: %s", err.Error()) - } else { - channel.Voucher = string(voucherJSON) - } - } - if channel.IsSender { - channel.IsInitiator = !channelState.IsPull() - channel.Transferred = channelState.Sent() - channel.OtherPeer = channelState.Recipient() - } else { - channel.IsInitiator = channelState.IsPull() - channel.Transferred = channelState.Received() - channel.OtherPeer = channelState.Sender() - } - return channel -} - type ExtendedPeerInfo struct { ID peer.ID Agent string @@ -179,24 +131,6 @@ type MessagePrototype struct { ValidNonce bool } -type RetrievalInfo struct { - PayloadCID cid.Cid - ID retrievalmarket.DealID - PieceCID *cid.Cid - PricePerByte abi.TokenAmount - UnsealPrice abi.TokenAmount - - Status retrievalmarket.DealStatus - Message string // more information about deal state, particularly errors - Provider peer.ID - BytesReceived uint64 - BytesPaidFor uint64 - TotalPaid abi.TokenAmount - - TransferChannelID *datatransfer.ChannelID - DataTransfer *DataTransferChannel -} - type SealingPipelineState struct { TaskJobsCount map[string]int MaxWaitDealsSectors uint64 diff --git a/build/openrpc/boost.json.gz b/build/openrpc/boost.json.gz index 3f2697b337cb2f977a64854607d95b9ac92582f7..4199a5b16f733f91b663013af0c2e52456a1c044 100644 GIT binary patch literal 3126 zcmV-649W8!iwFP!00000|Lk3RQ`@)}|0){Yf1qPR9+bA3=^p{wl3m*60^L{3+$#17 z(PB$pS*D~+zWW{dkvOrO*m+PMn(cIpBOOUc=lsrtl%!1I`19&Q1?e=d_D0d;=n&oZ&WZ)*7`Y9$DA}Ip9u!+h6W5Bc7V|p1NRj z%4!ELx32@q1$+RBE$l#G3+l_5#J6wXuJ~8h51Ea&0`wwU>|Bb_mfL7*>Xu(eHfIuU zW$iHjzlHn;vZ+%$q|P?l`f}MDO0W1W=TYg>bZTl-5eP!RieZv+GLeC>& zh}-yL00IyZfEbr(2oe!CL^k(4U@|~`E|4710L4tLVGH|25HDs*C=_#h!yTUoG~!|M z@f7p@Mvl=5o2GW6ey|7Nk?eIxyk!z~9bl67DTo}cL+T&_zJMUwF+l;i{hfGBkFu-y zo|r_$aRdYbIEap^)lgR=8lBSv2la-CpxwDk~_J=Lf z+Lygna1EaS55N7_4~N|I|LJ*s81`FxecB3#`(X%s&6_`HXpmGrQkN}TMKrUt1*utT3aT^b$^n>>9?o~9T zW^-@1pP<2xFZe%T%iW&R^WC1yd%GS9B#2$&(_LTi+u{GNfYq+}e@>e9Iv$Oz1(0%1 zf8@DX9#+)!#n3{TER8FQq8?l7O9M)>%VJP5DDgB=g0j5LjGryIAQpN(&82*b9xKig zG#--SK%?2pY(E+Cs6nLBSOg-+Bv=+ej!Cc>JbIkF<kWMQ?fH0&^?epjE(Kgz85qOH1 zLG4uV8|r{i6VD}C5U-?5=uv@Dw-P2~HoS2>7$3fzD_egR|F@(0EB#Now^Ty1^fHDn zXV)$*?2+4UOti6YVUM!RPw99f1R1WNi^aE}+xOg|iy3)`)%edLb#S{JQ!RWbKpmhX z)s%X@u@n8dsMp)kpTE@*mja7q=TvZ?2gFtDPW_vL4Ts$$_3w2+p%qh%y!&>e-fCL- zcMuBUok&#^{>K}`ZQQRnnyo!-;goOLXD6fwZp^iFNrZgob9WAEQi3Fi`5`7SFWajZGeSS`_7>SkX4qrr4)wJ*tkp#E%VcckB z;coC-zwc58lam5rp&!r59{v{<3FYUVZRfI&%Yel zf7~7)ygzRq2k-cYi(B#WPxilC=g33bI zweUIEl1B`DW8vTW{QzVta3{d7<3bB=ul2D{+HHGxD$9w-%!+ zFh`@R6=tU!^T46eKYElAF?`2t?x-Uur3zp_QjXl=Vg*>tyvbbNQ;tuBldGI|;iTMh zd7!IGVS=}1q$R=bs*7ap(cI7@cNZDEQ)Yh?dsBIf8e3A8M%J8_*{$BRRb6%366cjM zFQz)doi<;08kL_oCS_4#)|geh`ibv!mwKABfr{R*T*&vuUS{HCjuM-pVaJS7&|zTC z>T1YjJPRF()aaFwrdi8XW=?8iOX*g|T35Zx#K>wGFtPY$_K@{cD#L4mV5k1elGdU; zIF0SfG>JL-`6Yeys6+i$YeA@gFjRIPoYZ*6sjFq8e+{L@DZv%f0h%?}n<3C2$|Mn= zTN2SEE==OWBrZ(i!Xz$C;=&{@Oya^ME==OWBrZ(i!Xz#>l(^X2o0qtt0_>$L6~>c` z(i3R_HyIgU_cq#kkqzq3K%Iwj)@6OZIA1ZQw#F1i_ADFEN)+*>N?iLniUvx?wH?>x zHi$#_^p zr6@8kE*{!OTS-;e!f;r#-kcsEX)#{q5Lth$jR(u7zRHBrroPJ5SDE@MQ(t8gEGEJ7 zgcB^LzRJ{Bnfj`ym(`lQ-eX?-c`&d2Jkb2a>(|xt6LH3RnV0yAi7kGQZ-vxEvc^4S zC?+Q|FY!HPZ8os`NG|Gu!tR?lCm&=&HtE#hF{|S+F0B+S>ir$D0?0gwLN7`Hjw*BF6x0o zZfjm=>jYL#XduT-g%!%oW(bw}ULJ3Jnd6CNp?7ArxnJ6&S!QVz&Y#ji@Ke@o{a5s8 zBueAK>qnJw=MTQWeLLgEufN0jY!I+yT<}1#Z*N|!zljfZFkG$GUje1X=;uZg3(>D&xjFEsU8g$y6`#}K-*dx1tw`lqT{NmpO{ysr-B6+Er21>EZ6i94^FCxw z5D|^bB-2ui$&*M3mM=RUU&xa4>P3oc!J7=rD@RZezswGJUS5aKds;`QBoMo!h!nI60ma-LFCFp}Y!Rb%;9NEKyg z+G>}Yu~d2Uw5w90Ds6?y9rlZnNc=yb7 zRC*F&#{RtKt%(AvYx{%BL1c%kM;sa|3%}vrLVgp)EVq!~Lh5E#i$gTKSYJq@KcYI4 zI*2-f`EK}4d^h|=aQ$XJb}L^8K2rw1Y%$wpH?7=mlmwUBi=v*{L5fKRsh>Ggzbfc` z_Zw5tb3p9tu;SY%*_nmupyW-H!ZvqVBCEEwo7qrtc4u>~?B*te?i)}{U%ivo>w>;| zr_Mo0T_>t8eb2@CR|u6fpn%B2BD6wnC?L41VifQ5q3MXJIeTC=XU`2U-n_=6(dFp> Q0RRC1|1+fT8i|(x0FhZYlK=n! literal 5845 zcmV;`7AolLEJ!i1w189~qc?y7tENK00(XcU{M$$Yed1?60Vk_$4pgHp6@GD_MRe z-@bhVI`HSV4ZQ_82A^igLmKpuMjWaw5Y-@AXoh3kh)`dfIG#pl*w?~IZvh>+(1Uh3 zWpj^B{&?}qb$lGmaP)I(aokRB88TCt!iaq_X2^z&EYKaaT?;XIVAAe%dwp=Nv+lv` zpmpgvF7ha5EhfMXb3q{udb(TrYLtq3?6-Arl-^Z;lu;u7?ar*^{8x@|rdB z=N4rzjqpydKz}yb4HAxJ+*YE8g9rO-+gTw3f;G&jt zgX277uL!B-+%bHHW*Vz^n9k;7%|o_Bk!E3X19UL=EN~2Fl)C=$>(|quNy~8T*OQ3- zUb&vbCjIr8vDNFb<&0n3(5J|I4PE@&^_;uK|4k8TO`ZR_=pGz`_W*R0YW^~TpVb3j!4fg9R7aBLnT)l;H(Vhm0sX^@nYc+%Ztd-`w zd+?gg8=Ne<UVGZV3}Z9pL&=~H;_jmjpgQc^;klL{!Og*Kcmx`LH;vC|NGl-pe}xoRHPgpNr;r{ zhC!}-xoE+JO3M|oCD5)dP)ZfQk`lWqB{0eB%<$QE?2}Hnw_A2euiY*aBJZJJSAe{S ze!1`%JB}qMJ>El~FFC^5tR+1{6qV$7ZjyshU2bNhhvl{fds6}%)(=7Qv?W2OBaj5F z^EsRb>~fBL>N$(i40>iosUQL)k`Un%?}d_qD;Pu&`9h+CbSOY|>yjd(Y$*#jY^&Wu zYmRDLw><9nIJffL5j{-Etd$S;dZs_| zXL!&(fa9Ni+#d|~&B^?y*ZV&4`(qoL1M{wj@znedbTFD52J+VdxHoK}z%lqbZ?}6! zux!Cc1eVv=I=~b~9A6m5a}$s5W0*06#q7M)!(niqudrJE7OSM?k`p9MhzjU=aPhR6 zx7U+k-XXyhLks^zo9eC%q-NAt256%iERV2KV!Nos_V*?D-~#(JAcLw-9Uq94QX7(q zBdyZbu;o=7dqI80mKx1NC^&|0nR<)bJg{=3kTQG&41zBww-8%!Y$5PyR9%Q;?w8%y zbAy2de_vu`AfK@!!~~P+$|K)^-Gwkky3byP0ai@|sSrg$q-BBd#f73F4lUId2u0)F zy6WW2otz#${Sj_)tog7`urn9itjR-sF}MzcWrbg9fGjQY8VM(HCu@fSa5+nAx8&i$ zfrLD~LWXmTyiJY%B$#F_{v^ak&HW6j%GTdSTmMj$PLigCDP{mRC;kMSrl>ywvSIR1 zBdh3t7wP{(X;>v57p^F>p)D z4O;{)KD85URkv3|gtt!NtHH|=dHW#z{}t!2`2Xz~sy4hEhCn+59oU#8zBOgUidR2} zcVW}=iM$7_{pUG0!Li(=>fp>nYyc;eb!oRduYy0H+U?`u&p)h*g%Js8hk56Y4=wiI z)Mw`ZoDI*}=OzqoJ%kvfA{}?yy{-=aL>{w-&45*-`415T$8qkf4&r%BKD&TpWQDUf zqR^vnT*sI(u$^{C2XBa(elVhr2dC)L!v?xUUi2eqC!n7>mIW#D7$(6FAtIT_%lP>S zPYDc}=^lIvsH&@jb4VdL&gYyhmpT~we@rG8CTKlU1j*bD`y>Z{iIB(U9dx?qz26<@ zuXjK1e*XT?-?!-ff2sR!aN>KP|8r{mb@%@C?Wp_Sf8)ISbmx7zxc&XVbOQ8%U>MXZ z$Dz%w-jIx)IWd`#y@YhO(s_{{)4?YX5`Tic^^9a&gu@Glmigr}l0f;X(?Liv{VtpP z8N`GQkHOgQGa9;_k1g!a0zzdPO$T9yGW&j{gFhw{AJLfLE)bd0S^Vv6hKw86!HnEG z$*)C>^TzY?=wBfkFfp7{2k7AA>4GBv3KC2s%o<0|Mm?hM%k;!lYkv~-U|ua`b7(?}UQukLb+2^{FGc@c zyty)}!sKvRwL zg}iWD=@r4gSDrsB9t(S%u71BG%21A7;L`=u^T>>qCvSs6&im&;D$ifuFdV>nXR{Fk z{-9LG^K;91R)Gr@xKM!$6}V7=3l+FffeRJ5P=N~-xKM!$6}V7=i!B8%x`hWSv4@P> znF_1U!fc$#fm^o>mwQip@iOVuT_H9e>=3ye{S~wZLlnueY(H5?FfUd7+RuSEAnDhh zSe9eJV201ZaXYpNLh;^=mlJaNiD{=>OR3sH13p~xHBY$0RryF#{ei#`({ zK?|8@j=P9|8Qox)8T7GvU(Yv4O_!7*pBQ~r6Joubzf#q5=9;Gyt68jt+BZpqRTWx( z`FY6ZiGkaylHpiaJ#d3zLLrzBh7~oH5gI3kfn2)A@CrdckvSOgTvbL{0p(3>{0ga4 zGEZ2=_6pB!&ux2ASHs4AOBHNWsx^LLgPJulhKEzE+gGWasH6%EKWt!lJK@>~h{kT& zd{N;)75-D5$D!)qQSE>9em0zU-EGodV!vPkRU#0S^RDRX&MYTkg+|;q3UwG{2 zk;W$mMTKcOE_fKRu5*Y_46OVd-v*(H=p7Ffp;(Vd9pc+%ZdPLVi7eD36}tyT^@1~U z<$^6k)%z!faQIchYYNy9z&>5AAle`yDg#L7x_iz|8nVVp z=52-7jqolzscX07N2$cfa}y(6isIOsbuWsEZ;DA7oZG)NW~c5R+3yw=rCZwpr;E*s z(g`4yOFuu{0=A)dgJ|mmCSi_Qsr(bXus{F`-c(%#ipQ#C;B%9K0;1yFy&-BRlSQdg z8@H6&xXxEL`N#`0ID!SgLX%)I zqs#|>c4o8bqU*8)(c&dJq}$iQ75s5(ImS&Qy>oQ|kB~Z#8s!zh-@yBL1nsYp9%tu57~R^o2jRFTMc%xY38(n{D`a-qAuwir?2+LWnL+pQcP|I8Fyk|HTw z4wo;F!v&}yk);N(m4j0ekN*nBf22e^5Y&6)tx@DaHAJ#WXw4C966zMoHG{k&)(0Wh z1Hs|{^a_z?1uO?#6XePPyA^unqpt||K?wFxaBP~hL`@0T9I(cSb?c&)hx%iv0juab zOV>w&L&d4}{7i16SZPDF%?H>3ee(ftgT{qJPy+cO2zSsH970Vk(x1(eEe~7+M9Txa z4U!e0uL$-*2(}|Q7#S=>oR$@&Oj)>U(G@-nK52frmZ;Id2UvLGu}*acA>Lb|$rX;A zb91GKeUfM88af)ojr;Du&1w=hUwCO-$8A#DiPq?8YtcV~5d(*-Q zvIE$9mf5e{8+c}g00w#6i)fZh{UPi_n(B zY=||g1ZwNEhj0jrMRtZo4vV&u6SUA!EVu+>L+l|R60Xl20`wJY>VCN5N#sr z%mX8?6I@=PQwX#v^5D)G0fitq{B z$fwYD*Od_NT+EP)>mZ=-AD!q}f7J>G6_3?~L#@ywMpfoeYI-)LdZgx5vPZqs;n=8A z87*2FsS~;DRhDUBDUfd*-h_Xr| z3A$NrkwJPpYJ)TEysqFklB>iU)^|8v&{!FSMF^7W;Sw%vL})MrL21OG$ZGoGkQg3< zepte#n&i%bm#U^Adc9mJj^wDTO3u^-)s$|#AsY4t`6nTZ3~k0saq*5ik|7PDr1*N` zkbSAUng-mAtC~YLc2iXa^CBB;;4)__;R!3o|3PQSyF{B*Gmobr;9q8{jVPRkrdIsv%RTrQ zU-*!E{W(X3tqc2kh_QQdnfK4GS>EWvnTA+gIMe)lBtR^$m)Fb8?n?!eGeHa0iixGx zoL9eNO#!@Liz<94R-W_gdJQWKbKPlG_^e7fu;g-AFT9>0B+ht33aE!zKr|efDZ1G_ z?2s+t7IMr(zn2>jiKJSU;>WT@j8oD>D%=yxIH{5~J9R$g;6RX0xz?*gVmM4a(K^{M zLNFqeqsHx1!gmxJ9!*?6RW!OSnIE#(8x^&bGDB2dV|Bxky5UIOaP*{aII3%1tILk+ zZ5*k~j?`sGss#G8FM+NuJ5rY&z@D=df>V&9K+ z@W*80BN~T`E)bd0S@>;vod_tm*}x5iZ#UY1awxP$fx*mHCUPSJsBwfq7*05wdtQz% z!VBszJ%>7mW5stq)%8wOl=8?GBWFUSoV)atSN7Z_N19!isykk3K~>RueE6iX&Ag;o z08|M>gq&S(ldMWGAeIL*ZDUCyWwx$FP`z+Sb_cHR67YSjO>zL&Qj)h}E77d%?x8%~ z0;`_~yoQ;d8O2t126?FFW?axn_Wsm3;aY7BZs)?&>s9FM5_#A$1FXfHHL14Xc+g(` zS22BkBQ5lNqD)HxsmbREnR_sCJR8#R%s6^Qv5gAM(49E_ao84= zJ_*(K74q5KY+BEN1Li>r8gSm*%rO!EfK*8XRnkC}G-ysXbqY(ZjB1xO7>P<61QLkN zpWiyg(;=aO*vUNTX>g2NXETUNpapf!?FWx8p=ly>Wd-P(yU%-GDz!!hMG>kBs?|`F zvWa2Lu-mGPO3s?(9k8IxX0NAsi)7Ga#4;~((ge3PF&}t=4T$X%UfMi{(kd zBqq9h>v*fw(n=g408&ysq<|X)05@d}F`kc&M^uTCUsz&9$%~7_fuPLL)tpdlqtSfq zFZ^Kht&*h3buZW>4-TBHTNdq+=fIY2k!N66+9KzTfiFAqXob*xjn0tfuOHJHq=Ae< z1dc8Cc1%LCf{9NdF;MF(X=1-UGj9HVd%lrG4lD@$79a4-b60V!b8hrbC7>K02R(q)=hfYRtjN^ fdlSiqJGNJQ*aOSu_45A%00960DLl)n95n#|sW2#U diff --git a/cmd/boost/provider_cmd.go b/cmd/boost/provider_cmd.go index 966ba80f3..df2ff195f 100644 --- a/cmd/boost/provider_cmd.go +++ b/cmd/boost/provider_cmd.go @@ -5,17 +5,16 @@ import ( "sort" "strings" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/boost-gfm/storagemarket/network" bcli "github.com/filecoin-project/boost/cli" clinode "github.com/filecoin-project/boost/cli/node" "github.com/filecoin-project/boost/cmd" "github.com/filecoin-project/boost/extern/boostd-data/shared/cliutil" "github.com/filecoin-project/boost/retrievalmarket/lp2pimpl" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes/network" + "github.com/filecoin-project/go-address" "github.com/filecoin-project/lotus/chain/types" lcli "github.com/filecoin-project/lotus/cli" - "github.com/ipfs/go-cid" "github.com/ipni/go-libipni/maurl" "github.com/multiformats/go-multiaddr" "github.com/urfave/cli/v2" @@ -32,7 +31,6 @@ var providerCmd = &cli.Command{ Subcommands: []*cli.Command{ libp2pInfoCmd, storageAskCmd, - retrievalAskCmd, retrievalTransportsCmd, }, } @@ -209,100 +207,6 @@ var storageAskCmd = &cli.Command{ }, } -var retrievalAskCmd = &cli.Command{ - Name: "retrieval-ask", - Usage: "Query a storage provider's retrieval ask", - ArgsUsage: "[provider] [data CID]", - Flags: []cli.Flag{ - &cli.Int64Flag{ - Name: "size", - Usage: "data size in bytes", - }, - }, - Action: func(cctx *cli.Context) error { - ctx := bcli.ReqContext(cctx) - - afmt := NewAppFmt(cctx.App) - if cctx.NArg() != 2 { - afmt.Println("Usage: retrieval-ask [provider] [data CID]") - return nil - } - - n, err := clinode.Setup(cctx.String(cmd.FlagRepo.Name)) - if err != nil { - return err - } - - api, closer, err := lcli.GetGatewayAPI(cctx) - if err != nil { - return fmt.Errorf("cant setup gateway connection: %w", err) - } - defer closer() - - maddr, err := address.NewFromString(cctx.Args().First()) - if err != nil { - return err - } - - dataCid, err := cid.Parse(cctx.Args().Get(1)) - if err != nil { - return fmt.Errorf("parsing data cid: %w", err) - } - - addrInfo, err := cmd.GetAddrInfo(ctx, api, maddr) - if err != nil { - return err - } - - log.Debugw("found storage provider", "id", addrInfo.ID, "multiaddrs", addrInfo.Addrs, "addr", maddr) - - if err := n.Host.Connect(ctx, *addrInfo); err != nil { - return fmt.Errorf("failed to connect to peer %s: %w", addrInfo.ID, err) - } - - s, err := n.Host.NewStream(ctx, addrInfo.ID, QueryProtocolID) - if err != nil { - return fmt.Errorf("failed to open stream to peer %s: %w", addrInfo.ID, err) - } - defer s.Close() - - req := retrievalmarket.Query{ - PayloadCID: dataCid, - QueryParams: retrievalmarket.QueryParams{}, - } - - var ask retrievalmarket.QueryResponse - - if err := doRpc(ctx, s, &req, &ask); err != nil { - return fmt.Errorf("send retrieval-ask request rpc: %w", err) - } - - afmt.Printf("Status: %d\n", ask.Status) - if ask.Status != 0 { - return nil - } - afmt.Printf("Ask: %s\n", maddr) - afmt.Printf("Unseal price: %s\n", types.FIL(ask.UnsealPrice)) - afmt.Printf("Price per byte: %s\n", types.FIL(ask.MinPricePerByte)) - afmt.Printf("Payment interval: %s\n", types.SizeStr(types.NewInt(ask.MaxPaymentInterval))) - afmt.Printf("Payment interval increase: %s\n", types.SizeStr(types.NewInt(ask.MaxPaymentIntervalIncrease))) - - size := cctx.Uint64("size") - if size == 0 { - if ask.Size == 0 { - return nil - } - size = ask.Size - afmt.Printf("Size: %s\n", types.SizeStr(types.NewInt(ask.Size))) - } - transferPrice := types.BigMul(ask.MinPricePerByte, types.NewInt(size)) - totalPrice := types.BigAdd(ask.UnsealPrice, transferPrice) - afmt.Printf("Total price for %d bytes: %s\n", size, types.FIL(totalPrice)) - - return nil - }, -} - var retrievalTransportsCmd = &cli.Command{ Name: "retrieval-transports", Usage: "Query a storage provider's available retrieval transports (libp2p, http, etc)", diff --git a/cmd/boostd/dagstore.go b/cmd/boostd/dagstore.go deleted file mode 100644 index 2a44308f0..000000000 --- a/cmd/boostd/dagstore.go +++ /dev/null @@ -1,357 +0,0 @@ -package main - -import ( - "fmt" - "os" - "strings" - - "github.com/filecoin-project/lotus/lib/tablewriter" - "github.com/ipfs/go-cid" - - "github.com/fatih/color" - bapi "github.com/filecoin-project/boost/api" - bcli "github.com/filecoin-project/boost/cli" - lcli "github.com/filecoin-project/lotus/cli" - "github.com/urfave/cli/v2" -) - -var dagstoreCmd = &cli.Command{ - Name: "dagstore", - Usage: "Manage the dagstore on the Boost subsystem", - Subcommands: []*cli.Command{ - dagstoreRegisterShardCmd, - dagstoreInitializeShardCmd, - dagstoreRecoverShardCmd, - dagstoreInitializeAllCmd, - dagstoreListShardsCmd, - dagstoreGcCmd, - dagstoreDestroyShardCmd, - dagstoreLookupCmd, - }, -} - -var dagstoreGcCmd = &cli.Command{ - Name: "gc", - Usage: "Garbage collect the dagstore", - Flags: []cli.Flag{}, - Action: func(cctx *cli.Context) error { - ctx := lcli.ReqContext(cctx) - napi, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - collected, err := napi.BoostDagstoreGC(ctx) - if err != nil { - return err - } - - if len(collected) == 0 { - _, _ = fmt.Fprintln(os.Stdout, "no shards collected") - return nil - } - - for _, e := range collected { - if e.Error == "" { - _, _ = fmt.Fprintln(os.Stdout, e.Key, color.New(color.FgGreen).Sprint("SUCCESS")) - } else { - _, _ = fmt.Fprintln(os.Stdout, e.Key, color.New(color.FgRed).Sprint("ERROR"), e.Error) - } - } - - return nil - }, -} - -var dagstoreListShardsCmd = &cli.Command{ - Name: "list-shards", - Usage: "List all shards known to the dagstore, with their current status", - Flags: []cli.Flag{}, - Action: func(cctx *cli.Context) error { - ctx := lcli.ReqContext(cctx) - napi, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - shards, err := napi.BoostDagstoreListShards(ctx) - if err != nil { - return err - } - - return printTableShards(shards) - }, -} - -func printTableShards(shards []bapi.DagstoreShardInfo) error { - if len(shards) == 0 { - return nil - } - - tw := tablewriter.New( - tablewriter.Col("Key"), - tablewriter.Col("State"), - tablewriter.Col("Error"), - ) - - colors := map[string]color.Attribute{ - "ShardStateAvailable": color.FgGreen, - "ShardStateServing": color.FgBlue, - "ShardStateErrored": color.FgRed, - "ShardStateNew": color.FgYellow, - } - - for _, s := range shards { - m := map[string]interface{}{ - "Key": s.Key, - "State": func() string { - trimmedState := strings.TrimPrefix(s.State, "ShardState") - if c, ok := colors[s.State]; ok { - return color.New(c).Sprint(trimmedState) - } - return trimmedState - }(), - "Error": s.Error, - } - tw.Write(m) - } - return tw.Flush(os.Stdout) -} - -var dagstoreRegisterShardCmd = &cli.Command{ - Name: "register-shard", - ArgsUsage: "[key]", - Usage: "Register a shard", - Flags: []cli.Flag{ - &cli.BoolFlag{ - Name: "color", - Usage: "use color in display output", - DefaultText: "depends on output being a TTY", - }, - }, - Action: func(cctx *cli.Context) error { - if cctx.IsSet("color") { - color.NoColor = !cctx.Bool("color") - } - - if cctx.NArg() != 1 { - return fmt.Errorf("must provide a single shard key") - } - - napi, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - ctx := lcli.ReqContext(cctx) - - shardKey := cctx.Args().First() - err = napi.BoostDagstoreRegisterShard(ctx, shardKey) - if err != nil { - return err - } - - fmt.Println("Registered shard " + shardKey) - return nil - }, -} - -var dagstoreInitializeShardCmd = &cli.Command{ - Name: "initialize-shard", - ArgsUsage: "[key]", - Usage: "Initialize the specified shard", - Flags: []cli.Flag{}, - Action: func(cctx *cli.Context) error { - if cctx.NArg() != 1 { - return fmt.Errorf("must provide a single shard key") - } - - ctx := lcli.ReqContext(cctx) - napi, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - return napi.BoostDagstoreInitializeShard(ctx, cctx.Args().First()) - }, -} - -var dagstoreInitializeAllCmd = &cli.Command{ - Name: "initialize-all", - Usage: "Initialize all uninitialized shards, streaming results as they're produced; only shards for unsealed pieces are initialized by default", - Flags: []cli.Flag{ - &cli.UintFlag{ - Name: "concurrency", - Usage: "maximum shards to initialize concurrently at a time; use 0 for unlimited", - Required: true, - }, - &cli.BoolFlag{ - Name: "include-sealed", - Usage: "initialize sealed pieces as well", - }, - }, - Action: func(cctx *cli.Context) error { - concurrency := cctx.Uint("concurrency") - sealed := cctx.Bool("include-sealed") - - ctx := lcli.ReqContext(cctx) - - napi, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - params := bapi.DagstoreInitializeAllParams{ - MaxConcurrency: int(concurrency), - IncludeSealed: sealed, - } - - ch, err := napi.BoostDagstoreInitializeAll(ctx, params) - if err != nil { - return err - } - - for { - select { - case evt, ok := <-ch: - if !ok { - return nil - } - _, _ = fmt.Fprint(os.Stdout, color.New(color.BgHiBlack).Sprintf("(%d/%d)", evt.Current, evt.Total)) - _, _ = fmt.Fprint(os.Stdout, " ") - if evt.Event == "start" { - _, _ = fmt.Fprintln(os.Stdout, evt.Key, color.New(color.Reset).Sprint("STARTING")) - } else { - if evt.Success { - _, _ = fmt.Fprintln(os.Stdout, evt.Key, color.New(color.FgGreen).Sprint("SUCCESS")) - } else { - _, _ = fmt.Fprintln(os.Stdout, evt.Key, color.New(color.FgRed).Sprint("ERROR"), evt.Error) - } - } - - case <-ctx.Done(): - return fmt.Errorf("aborted") - } - } - }, -} - -var dagstoreRecoverShardCmd = &cli.Command{ - Name: "recover-shard", - ArgsUsage: "[key]", - Usage: "Attempt to recover a shard in errored state", - Flags: []cli.Flag{ - &cli.BoolFlag{ - Name: "color", - Usage: "use color in display output", - DefaultText: "depends on output being a TTY", - }, - }, - Action: func(cctx *cli.Context) error { - if cctx.IsSet("color") { - color.NoColor = !cctx.Bool("color") - } - - if cctx.NArg() != 1 { - return fmt.Errorf("must provide a single shard key") - } - - napi, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - ctx := lcli.ReqContext(cctx) - - return napi.BoostDagstoreRecoverShard(ctx, cctx.Args().First()) - }, -} - -var dagstoreDestroyShardCmd = &cli.Command{ - Name: "destroy-shard", - ArgsUsage: "[key]", - Usage: "Destroy a shard", - Flags: []cli.Flag{ - &cli.BoolFlag{ - Name: "color", - Usage: "use color in display output", - DefaultText: "depends on output being a TTY", - }, - }, - Action: func(cctx *cli.Context) error { - if cctx.IsSet("color") { - color.NoColor = !cctx.Bool("color") - } - - if cctx.NArg() != 1 { - return fmt.Errorf("must provide a single shard key") - } - - napi, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - ctx := lcli.ReqContext(cctx) - - shardKey := cctx.Args().First() - err = napi.BoostDagstoreDestroyShard(ctx, shardKey) - if err != nil { - return err - } - - fmt.Println("Destroyed shard " + shardKey) - return nil - }, -} - -var dagstoreLookupCmd = &cli.Command{ - Name: "lookup-piece-cid", - ArgsUsage: "[key]", - Usage: "Performs a reverse lookup with payload CID to get Piece CID", - Aliases: []string{"lpc"}, - Flags: []cli.Flag{ - &cli.BoolFlag{ - Name: "color", - Usage: "use color in display output", - DefaultText: "depends on output being a TTY", - }, - }, - Action: func(cctx *cli.Context) error { - if cctx.IsSet("color") { - color.NoColor = !cctx.Bool("color") - } - - if cctx.NArg() != 1 { - return fmt.Errorf("must provide a single payload CID") - } - - napi, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - ctx := lcli.ReqContext(cctx) - - shardKey := cctx.Args().First() - payloadCid, err := cid.Parse(shardKey) - if err != nil { - return fmt.Errorf("Unable to parse the provided CID: %s", shardKey) - } - pieceCid, err := napi.BoostDagstorePiecesContainingMultihash(ctx, payloadCid.Hash()) - if err != nil { - return err - } - - fmt.Printf("Given CID was found in the following pieces: %s", pieceCid) - return nil - }, -} diff --git a/cmd/boostd/import_data.go b/cmd/boostd/import_data.go index bed8bcb33..35adfa617 100644 --- a/cmd/boostd/import_data.go +++ b/cmd/boostd/import_data.go @@ -77,19 +77,7 @@ var importDataCmd = &cli.Command{ return err } - if deleteAfterImport { - return fmt.Errorf("cannot find boost deal with proposal cid %s and legacy deal data cannot be automatically deleted after import (only new deals)", proposalCid) - } - - // The deal is not in the boost database, try the legacy - // markets datastore (v1.1.0 deal) - err := napi.MarketImportDealData(cctx.Context, *proposalCid, filePath) - if err != nil { - return fmt.Errorf("couldnt import v1.1.0 deal, or find boost deal: %w", err) - } - - fmt.Printf("Offline deal import for v1.1.0 deal %s scheduled for execution\n", proposalCid.String()) - return nil + return fmt.Errorf("cannot find boost deal with proposal cid %s and legacy deals are no olnger supported", proposalCid) } // Get the deal UUID from the deal diff --git a/cmd/boostd/index.go b/cmd/boostd/index.go index 759c01012..e0c575a33 100644 --- a/cmd/boostd/index.go +++ b/cmd/boostd/index.go @@ -262,11 +262,15 @@ var indexProvAnnounceDeal = &cli.Command{ return nil } // Announce legacy deal - err = napi.BoostIndexerAnnounceLegacyDeal(ctx, proposalCid) + ad, err := napi.BoostIndexerAnnounceLegacyDeal(ctx, proposalCid) if err != nil { return fmt.Errorf("announcing legacy deal with proposal CID %s: %w", proposalCid, err) } - fmt.Printf("Announced the legacy deal") + if ad.Defined() { + fmt.Printf("Announced the legacy deal with Ad cid %s\n", ad) + return nil + } + fmt.Printf("Legacy deal already announced\n") return nil }, } diff --git a/cmd/boostd/init.go b/cmd/boostd/init.go index a3563ed2a..b61408cbd 100644 --- a/cmd/boostd/init.go +++ b/cmd/boostd/init.go @@ -1,36 +1,24 @@ package main import ( - "bufio" "context" "errors" "fmt" - "io" "os" - "os/exec" "path" "strings" - "github.com/chzyer/readline" - "github.com/dustin/go-humanize" - "github.com/filecoin-project/boost/api" cliutil "github.com/filecoin-project/boost/cli/util" scliutil "github.com/filecoin-project/boost/extern/boostd-data/shared/cliutil" "github.com/filecoin-project/boost/node/config" - "github.com/filecoin-project/boost/node/impl/backupmgr" "github.com/filecoin-project/boost/node/repo" - "github.com/filecoin-project/boost/util" "github.com/filecoin-project/go-address" lapi "github.com/filecoin-project/lotus/api" "github.com/filecoin-project/lotus/api/client" "github.com/filecoin-project/lotus/api/v0api" lcli "github.com/filecoin-project/lotus/cli" - lotus_config "github.com/filecoin-project/lotus/node/config" - lotus_modules "github.com/filecoin-project/lotus/node/modules" lotus_repo "github.com/filecoin-project/lotus/node/repo" - "github.com/gbrlsnchs/jwt/v3" "github.com/ipfs/go-datastore" - dsq "github.com/ipfs/go-datastore/query" "github.com/urfave/cli/v2" ) @@ -151,395 +139,6 @@ var initCmd = &cli.Command{ }, } -var migrateFlags = []cli.Flag{ - &cli.StringFlag{ - Name: "wallet-publish-storage-deals", - Usage: "wallet to be used for PublishStorageDeals messages", - Required: true, - }, - &cli.StringFlag{ - Name: "wallet-deal-collateral", - Usage: "wallet to be used for deal collateral", - Required: true, - }, - &cli.Int64Flag{ - Name: "max-staging-deals-bytes", - Usage: "max size for staging area in bytes", - Required: true, - }, -} - -var migrateMarketsCmd = &cli.Command{ - Name: "migrate-markets", - Usage: "Migrate from an existing split markets (MRA) repo to Boost", - Flags: append([]cli.Flag{ - &cli.StringFlag{ - Name: "import-markets-repo", - Usage: "initialize boost from an existing split markets (MRA) repo", - Required: true, - }}, - migrateFlags..., - ), - Before: before, - Action: func(cctx *cli.Context) error { - return migrate(cctx, false, cctx.String("import-markets-repo")) - }, -} - -var migrateMonolithCmd = &cli.Command{ - Name: "migrate-monolith", - Usage: "Migrate from an existing monolith lotus-miner repo to Boost", - Flags: append([]cli.Flag{ - &cli.StringFlag{ - Name: "import-miner-repo", - Usage: "initialize boost from an existing monolith lotus-miner repo", - Required: true, - }}, - append(minerApiFlags, migrateFlags...)..., - ), - Before: before, - Action: func(cctx *cli.Context) error { - return migrate(cctx, true, cctx.String("import-miner-repo")) - }, -} - -func migrate(cctx *cli.Context, fromMonolith bool, mktsRepoPath string) error { - ctx := scliutil.ReqContext(cctx) - - // Open markets repo - fmt.Printf("Opening repo '%s'\n", mktsRepoPath) - mktsRepo, err := getMarketsRepo(mktsRepoPath) - if err != nil { - return err - } - defer mktsRepo.Close() //nolint:errcheck - - // Initialize boost repo - bp, err := initBoost(ctx, cctx, mktsRepo) - if err != nil { - return err - } - - boostRepo, err := bp.repo.Lock(repo.Boost) - if err != nil { - return err - } - defer boostRepo.Close() - - ds, err := boostRepo.Datastore(context.Background(), metadataNamespace) - if err != nil { - return err - } - - // Migrate datastore keys - fmt.Println("Migrating datastore keys") - err = migrateMarketsDatastore(ctx, ds, mktsRepo) - if err != nil { - return err - } - - // Migrate keystore - fmt.Println("Migrating keystore") - err = backupmgr.CopyKeysBetweenRepos(mktsRepo, boostRepo) - if err != nil { - return err - } - - // Migrate config - fmt.Println("Migrating markets config") - err = migrateMarketsConfig(cctx, mktsRepo, boostRepo, bp, fromMonolith) - if err != nil { - return err - } - - // Add the miner address to the metadata datastore - fmt.Printf("Adding miner address %s to datastore\n", bp.minerActor) - err = addMinerAddressToDatastore(ds, bp.minerActor) - if err != nil { - return err - } - - // Copy the storage.json file if there is one, otherwise create an empty one - err = migrateStorageJson(mktsRepo.Path(), boostRepo.Path()) - if err != nil { - return err - } - - // Create an auth token - err = createAuthToken(bp.repo, boostRepo) - if err != nil { - return err - } - - // Migrate DAG store - err = migrateDAGStore(ctx, mktsRepo, boostRepo) - if err != nil { - return err - } - - fmt.Println("Boost repo successfully created at " + boostRepo.Path()) - fmt.Println("You can now start boost with 'boostd -vv run'") - - return nil -} - -func migrateStorageJson(mktsRepoPath string, boostRepoPath string) error { - mktsFilePath := path.Join(mktsRepoPath, "storage.json") - boostFilePath := path.Join(boostRepoPath, "storage.json") - - // Read storage.json in the markets repo - bz, err := os.ReadFile(mktsFilePath) - if err != nil { - if !errors.Is(err, os.ErrNotExist) { - return fmt.Errorf("reading %s: %w", mktsFilePath, err) - } - - // There is no storage.json in the markets repo, so create an empty one - // in the Boost repo - fmt.Println("Creating storage.json file") - bz = []byte("{}") - } else { - fmt.Println("Migrating storage.json file") - } - - // Write storage.json in the boost repo - err = os.WriteFile(boostFilePath, bz, 0666) - if err != nil { - return fmt.Errorf("writing %s: %w", boostFilePath, err) - } - - return nil -} - -func createAuthToken(boostRepo *lotus_repo.FsRepo, boostRepoLocked lotus_repo.LockedRepo) error { - ks, err := boostRepoLocked.KeyStore() - if err != nil { - return fmt.Errorf("getting boost keystore: %w", err) - } - - // Set up the API secret key in the keystore - _, err = lotus_modules.APISecret(ks, boostRepoLocked) - if err != nil { - return fmt.Errorf("generating API token: %w", err) - } - - // Get the API token from the repo - _, err = boostRepo.APIToken() - if err == nil { - // If the token already exists, nothing more to do - return nil - } - - // Check if the error was because the token has not been created (expected) - // or for some other reason - if !errors.Is(err, lotus_repo.ErrNoAPIEndpoint) { - return fmt.Errorf("getting API token for newly created boost repo: %w", err) - } - - // The token does not exist, so create a new token - p := lotus_modules.JwtPayload{ - Allow: api.AllPermissions, - } - - // Get the API secret key - key, err := ks.Get(lotus_modules.JWTSecretName) - if err != nil { - // This should never happen because it gets created by the APISecret - // function above - return fmt.Errorf("getting key %s from keystore to generate API token: %w", - lotus_modules.JWTSecretName, err) - } - - // Create the API token - cliToken, err := jwt.Sign(&p, jwt.NewHS256(key.PrivateKey)) - if err != nil { - return fmt.Errorf("signing JSW payload for API token: %w", err) - } - - // Save the API token in the repo - err = boostRepoLocked.SetAPIToken(cliToken) - if err != nil { - return fmt.Errorf("setting boost API token: %w", err) - } - - return nil -} - -func migrateDAGStore(ctx context.Context, mktsRepo lotus_repo.LockedRepo, boostRepo lotus_repo.LockedRepo) error { - - subdir := "dagstore" - - mktsSubdirPath := path.Join(mktsRepo.Path(), subdir) - boostSubdirPath := path.Join(boostRepo.Path(), subdir) - - rawMktsCfg, err := mktsRepo.Config() - if err != nil { - return fmt.Errorf("getting markets repo config: %w", err) - } - mktsCfg, ok := rawMktsCfg.(*lotus_config.StorageMiner) - if !ok { - return fmt.Errorf("expected legacy markets config, got %T", rawMktsCfg) - } - - if len(mktsCfg.DAGStore.RootDir) > 0 { - fmt.Println("Not migrating the dagstore as a custom dagstore path is set. Please manually move or copy the dagstore to $BOOST_PATH/dagstore") - return nil - } - - dirInfo, err := os.Lstat(mktsSubdirPath) - if err != nil { - if errors.Is(err, os.ErrNotExist) { - return nil - } - return fmt.Errorf("reading %s path %s", subdir, mktsSubdirPath) - } - - // If it's a sym-link just copy the sym-link - if dirInfo.Mode()&os.ModeSymlink == os.ModeSymlink { - fmt.Printf("copying sym-link %s to %s\n", mktsSubdirPath, boostSubdirPath) - cmd := exec.Command("cp", "-a", mktsSubdirPath, boostSubdirPath) - err = cmd.Run() - if err != nil { - return fmt.Errorf("Copying sym-link %s %s to %s: %w", subdir, mktsSubdirPath, boostSubdirPath, err) - } - return nil - } - - if !dirInfo.IsDir() { - return fmt.Errorf("expected %s to be a directory but it's not", mktsSubdirPath) - } - - dirSizeBytes, err := util.DirSize(mktsSubdirPath) - if err != nil { - return fmt.Errorf("getting size of %s: %w", mktsSubdirPath, err) - } - - humanSize := humanize.Bytes(uint64(dirSizeBytes)) - fmt.Printf("%s directory size: %s\n", subdir, humanSize) - - // If the directory is small enough, just copy it - if dirSizeBytes < 1024 { - fmt.Printf("Copying %s to %s\n", mktsSubdirPath, boostSubdirPath) - cmd := exec.Command("cp", "-r", mktsSubdirPath, boostSubdirPath) - err = cmd.Run() - if err != nil { - return fmt.Errorf("Copying %s directory %s to %s: %w", subdir, mktsSubdirPath, boostSubdirPath, err) - } - return nil - } - - cs := readline.NewCancelableStdin(os.Stdin) - go func() { - <-ctx.Done() - cs.Close() // nolint:errcheck - }() - rl := bufio.NewReader(cs) - for { - fmt.Printf("%s directory size is %s. Copy [c] / Move [m] / Ignore [i]:\n", subdir, humanSize) - - line, _, err := rl.ReadLine() - if err != nil { - if errors.Is(err, io.EOF) { - return fmt.Errorf("boost initialize canceled: %w", err) - } - - return fmt.Errorf("reading input: %w", err) - } - - switch string(line) { - case "c", "y": - fmt.Printf("Copying %s to %s\n", mktsSubdirPath, boostSubdirPath) - cmd := exec.Command("cp", "-r", mktsSubdirPath, boostSubdirPath) - err = cmd.Run() - if err != nil { - return fmt.Errorf("Copying %s directory %s to %s: %w", subdir, mktsSubdirPath, boostSubdirPath, err) - } - return nil - case "m": - fmt.Printf("Moving %s to %s\n", mktsSubdirPath, boostSubdirPath) - cmd := exec.Command("mv", mktsSubdirPath, boostSubdirPath) - err = cmd.Run() - if err != nil { - return fmt.Errorf("Moving %s directory %s to %s: %w", subdir, mktsSubdirPath, boostSubdirPath, err) - } - return nil - case "i": - fmt.Printf("Not copying %s directory from markets to boost\n", subdir) - return nil - } - } -} - -func migrateMarketsConfig(cctx *cli.Context, mktsRepo lotus_repo.LockedRepo, boostRepo lotus_repo.LockedRepo, bp *boostParams, fromMonolith bool) error { - var cerr error - err := boostRepo.SetConfig(func(raw interface{}) { - rcfg, ok := raw.(*config.Boost) - if !ok { - cerr = errors.New("expected boost config") - return - } - - rawMktsCfg, err := mktsRepo.Config() - if err != nil { - cerr = fmt.Errorf("getting markets repo config: %w", err) - return - } - mktsCfg, ok := rawMktsCfg.(*lotus_config.StorageMiner) - if !ok { - cerr = fmt.Errorf("expected legacy markets config, got %T", rawMktsCfg) - return - } - - if !fromMonolith { - // When migrating from a split markets process, copy across the API - // listen address because we're going to replace the split markets - // process with boost. - // (When migrating from a monolith leave the defaults which are - // different from lotus miner, so they won't clash). - rcfg.Common.API = mktsCfg.Common.API - } - rcfg.Common.Backup = mktsCfg.Common.Backup - rcfg.Common.Libp2p = mktsCfg.Common.Libp2p - rcfg.Storage.ParallelFetchLimit = mktsCfg.Storage.ParallelFetchLimit - setBoostDealMakingCfg(&rcfg.Dealmaking, mktsCfg) - rcfg.LotusDealmaking = mktsCfg.Dealmaking - rcfg.LotusFees.MaxMarketBalanceAddFee = mktsCfg.Fees.MaxMarketBalanceAddFee - rcfg.LotusFees.MaxPublishDealsFee = mktsCfg.Fees.MaxPublishDealsFee - rcfg.DAGStore = mktsCfg.DAGStore - // Clear the DAG store root dir config, because the DAG store is no longer configurable in Boost - // (it is always at /dagstore - rcfg.DAGStore.RootDir = "" - rcfg.IndexProvider.EntriesCacheCapacity = mktsCfg.IndexProvider.EntriesCacheCapacity - rcfg.IndexProvider.EntriesChunkSize = mktsCfg.IndexProvider.EntriesChunkSize - rcfg.IndexProvider.TopicName = mktsCfg.IndexProvider.TopicName - rcfg.IndexProvider.PurgeCacheOnStart = mktsCfg.IndexProvider.PurgeCacheOnStart - rcfg.IndexProvider.Enable = true // Enable index provider in Boost by default - - if fromMonolith { - // If migrating from a monolith miner, read the sealing and - // indexing endpoints from the command line parameters - cerr = setMinerApiConfig(cctx, rcfg, false) - if cerr != nil { - return - } - } else { - // If migrating from a split markets process, just copy across - // the sealing and indexing endpoints. - rcfg.SealerApiInfo = mktsCfg.Subsystems.SealerApiInfo - rcfg.SectorIndexApiInfo = mktsCfg.Subsystems.SectorIndexApiInfo - } - setCommonConfig(cctx, rcfg, bp) - }) - if cerr != nil { - return cerr - } - if err != nil { - return fmt.Errorf("setting config: %w", err) - } - - return nil -} - type boostParams struct { repo *lotus_repo.FsRepo minerActor address.Address @@ -703,126 +302,6 @@ func addMinerAddressToDatastore(ds datastore.Batching, minerActor address.Addres return ds.Put(context.Background(), minerAddrDSKey, minerActor.Bytes()) } -func setBoostDealMakingCfg(bdm *config.DealmakingConfig, mktsCfg *lotus_config.StorageMiner) { - ldm := mktsCfg.Dealmaking - bdm.ConsiderOnlineStorageDeals = ldm.ConsiderOnlineStorageDeals - bdm.ConsiderOfflineStorageDeals = ldm.ConsiderOfflineStorageDeals - bdm.ConsiderOnlineRetrievalDeals = ldm.ConsiderOnlineRetrievalDeals - bdm.ConsiderOfflineRetrievalDeals = ldm.ConsiderOfflineRetrievalDeals - bdm.ConsiderVerifiedStorageDeals = ldm.ConsiderVerifiedStorageDeals - bdm.ConsiderUnverifiedStorageDeals = ldm.ConsiderUnverifiedStorageDeals - bdm.PieceCidBlocklist = ldm.PieceCidBlocklist - bdm.ExpectedSealDuration = config.Duration(ldm.ExpectedSealDuration) - bdm.MaxDealStartDelay = config.Duration(ldm.MaxDealStartDelay) - bdm.MaxProviderCollateralMultiplier = ldm.MaxProviderCollateralMultiplier - bdm.MaxStagingDealsBytes = ldm.MaxStagingDealsBytes - bdm.StartEpochSealingBuffer = ldm.StartEpochSealingBuffer - bdm.Filter = ldm.Filter - bdm.RetrievalFilter = ldm.RetrievalFilter - bdm.RetrievalPricing = ldm.RetrievalPricing -} - -func getMarketsRepo(repoPath string) (lotus_repo.LockedRepo, error) { - // Open the repo at the repo path - mktsRepo, err := lotus_repo.NewFS(repoPath) - if err != nil { - return nil, fmt.Errorf("opening legacy markets repo %s: %w", repoPath, err) - } - - // Make sure the repo exists - exists, err := mktsRepo.Exists() - if err != nil { - return nil, fmt.Errorf("checking legacy markets repo %s exists: %w", repoPath, err) - } - if !exists { - return nil, fmt.Errorf("legacy markets repo %s does not exist", repoPath) - } - - // Lock the repo - lr, err := mktsRepo.LockRO(lotus_repo.StorageMiner) - if err != nil { - return nil, fmt.Errorf("locking legacy markets repo %s: %w", repoPath, err) - } - return lr, nil -} - -func migrateMarketsDatastore(ctx context.Context, boostDS datastore.Batching, mktsRepo lotus_repo.LockedRepo) error { - // Open the metadata datastore on the repo - mktsDS, err := mktsRepo.Datastore(ctx, metadataNamespace) - if err != nil { - return fmt.Errorf("opening datastore %s on legacy markets repo %s: %w", - metadataNamespace, mktsRepo.Path(), err) - } - - // Import the key / values from the markets metadata datastore - prefixes := []string{ - // Storage deals - "/deals/provider", - // Retrieval deals - "/retrievals/provider", - // Piece store - "/storagemarket", - } - for _, prefix := range prefixes { - err := importPrefix(ctx, prefix, mktsDS, boostDS) - if err != nil { - return err - } - } - - return nil -} - -func importPrefix(ctx context.Context, prefix string, mktsDS datastore.Batching, boostDS datastore.Batching) error { - fmt.Printf("Importing all legacy markets datastore keys under %s\n", prefix) - - q, err := mktsDS.Query(ctx, dsq.Query{ - Prefix: prefix, - }) - if err != nil { - return fmt.Errorf("legacy markets datastore query: %w", err) - } - defer q.Close() //nolint:errcheck - - // Import keys in batches - totalCount := 0 - batchSize := 1024 - results := q.Next() - for { - batch, err := boostDS.Batch(ctx) - if err != nil { - return fmt.Errorf("creating boost datastore batch: %w", err) - } - - complete := false - count := 0 - for ; count < batchSize; count++ { - res, ok := <-results - if !ok { - complete = true - break - } - - err := batch.Put(ctx, datastore.NewKey(res.Key), res.Value) - if err != nil { - return fmt.Errorf("putting %s to Boost datastore: %w", res.Key, err) - } - } - - fmt.Printf("Importing %d legacy markets datastore keys\n", count) - err = batch.Commit(ctx) - if err != nil { - return fmt.Errorf("saving %d datastore keys to Boost datastore: %w", count, err) - } - - totalCount += count - if complete { - fmt.Printf("Imported %d legacy markets datastore keys under %s\n", totalCount, prefix) - return nil - } - } -} - // checkV1ApiSupport uses v0 api version to signal support for v1 API // trying to query the v1 api on older lotus versions would get a 404, which can happen for any number of other reasons func checkV1ApiSupport(ctx context.Context, cctx *cli.Context) error { diff --git a/cmd/boostd/legacy_data_transfers.go b/cmd/boostd/legacy_data_transfers.go deleted file mode 100644 index 425417140..000000000 --- a/cmd/boostd/legacy_data_transfers.go +++ /dev/null @@ -1,325 +0,0 @@ -package main - -import ( - "context" - "encoding/json" - "errors" - "fmt" - bapi "github.com/filecoin-project/boost/api" - datatransferv2 "github.com/filecoin-project/go-data-transfer/v2" - "github.com/filecoin-project/lotus/api" - "os" - "strconv" - "time" - - tm "github.com/buger/goterm" - bcli "github.com/filecoin-project/boost/cli" - datatransfer "github.com/filecoin-project/go-data-transfer" - lcli "github.com/filecoin-project/lotus/cli" - "github.com/libp2p/go-libp2p/core/peer" - "github.com/urfave/cli/v2" -) - -var dataTransfersCmd = &cli.Command{ - Name: "data-transfers", - Usage: "Manage legacy data transfers (Markets V1)", - Category: "legacy", - Subcommands: []*cli.Command{ - transfersListCmd, - marketRestartTransfer, - marketCancelTransfer, - transfersDiagnosticsCmd, - }, -} - -var marketRestartTransfer = &cli.Command{ - Name: "restart", - Usage: "Force restart a stalled data transfer", - ArgsUsage: "", - Flags: []cli.Flag{ - &cli.StringFlag{ - Name: "peerid", - Usage: "narrow to transfer with specific peer", - }, - &cli.BoolFlag{ - Name: "initiator", - Usage: "specify only transfers where peer is/is not initiator", - Value: false, - }, - }, - Action: func(cctx *cli.Context) error { - if !cctx.Args().Present() { - return cli.ShowCommandHelp(cctx, cctx.Command.Name) - } - nodeApi, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - ctx := lcli.ReqContext(cctx) - - transferUint, err := strconv.ParseUint(cctx.Args().First(), 10, 64) - if err != nil { - return fmt.Errorf("Error reading transfer ID: %w", err) - } - transferID := datatransfer.TransferID(transferUint) - initiator := cctx.Bool("initiator") - var other peer.ID - if pidstr := cctx.String("peerid"); pidstr != "" { - p, err := peer.Decode(pidstr) - if err != nil { - return err - } - other = p - } else { - channels, err := nodeApi.MarketListDataTransfers(ctx) - if err != nil { - return err - } - found := false - for _, channel := range channels { - if channel.IsInitiator == initiator && channel.TransferID == transferID { - other = channel.OtherPeer - found = true - break - } - } - if !found { - return errors.New("unable to find matching data transfer") - } - } - - return nodeApi.MarketRestartDataTransfer(ctx, transferID, other, initiator) - }, -} - -var marketCancelTransfer = &cli.Command{ - Name: "cancel", - Usage: "Force cancel a data transfer", - ArgsUsage: "", - Flags: []cli.Flag{ - &cli.StringFlag{ - Name: "peerid", - Usage: "narrow to transfer with specific peer", - }, - &cli.BoolFlag{ - Name: "initiator", - Usage: "specify only transfers where peer is/is not initiator", - Value: false, - }, - &cli.DurationFlag{ - Name: "cancel-timeout", - Usage: "time to wait for cancel to be sent to client", - Value: 5 * time.Second, - }, - }, - Action: func(cctx *cli.Context) error { - if !cctx.Args().Present() { - return cli.ShowCommandHelp(cctx, cctx.Command.Name) - } - nodeApi, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - ctx := lcli.ReqContext(cctx) - - transferUint, err := strconv.ParseUint(cctx.Args().First(), 10, 64) - if err != nil { - return fmt.Errorf("Error reading transfer ID: %w", err) - } - transferID := datatransfer.TransferID(transferUint) - initiator := cctx.Bool("initiator") - var other peer.ID - if pidstr := cctx.String("peerid"); pidstr != "" { - p, err := peer.Decode(pidstr) - if err != nil { - return err - } - other = p - } else { - channels, err := nodeApi.MarketListDataTransfers(ctx) - if err != nil { - return err - } - found := false - for _, channel := range channels { - if channel.IsInitiator == initiator && channel.TransferID == transferID { - other = channel.OtherPeer - found = true - break - } - } - if !found { - return errors.New("unable to find matching data transfer") - } - } - - timeoutCtx, cancel := context.WithTimeout(ctx, cctx.Duration("cancel-timeout")) - defer cancel() - return nodeApi.MarketCancelDataTransfer(timeoutCtx, transferID, other, initiator) - }, -} - -var transfersListCmd = &cli.Command{ - Name: "list", - Usage: "List ongoing data transfers for this miner", - Flags: []cli.Flag{ - &cli.BoolFlag{ - Name: "verbose", - Aliases: []string{"v"}, - Usage: "print verbose transfer details", - }, - &cli.BoolFlag{ - Name: "completed", - Usage: "show completed data transfers", - }, - &cli.BoolFlag{ - Name: "watch", - Usage: "watch deal updates in real-time, rather than a one time list", - }, - &cli.BoolFlag{ - Name: "show-failed", - Usage: "show failed/cancelled transfers", - }, - }, - Action: func(cctx *cli.Context) error { - api, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - ctx := lcli.ReqContext(cctx) - - channels, err := api.MarketListDataTransfers(ctx) - if err != nil { - return err - } - - verbose := cctx.Bool("verbose") - completed := cctx.Bool("completed") - watch := cctx.Bool("watch") - showFailed := cctx.Bool("show-failed") - if watch { - channelUpdates, err := api.MarketDataTransferUpdates(ctx) - if err != nil { - return err - } - - for { - tm.Clear() // Clear current screen - - tm.MoveCursor(1, 1) - - lcli.OutputDataTransferChannels(tm.Screen, toDTv2Channels(channels), verbose, completed, showFailed) - - tm.Flush() - - select { - case <-ctx.Done(): - return nil - case channelUpdate := <-channelUpdates: - var found bool - for i, existing := range channels { - if existing.TransferID == channelUpdate.TransferID && - existing.OtherPeer == channelUpdate.OtherPeer && - existing.IsSender == channelUpdate.IsSender && - existing.IsInitiator == channelUpdate.IsInitiator { - channels[i] = channelUpdate - found = true - break - } - } - if !found { - channels = append(channels, channelUpdate) - } - } - } - } - lcli.OutputDataTransferChannels(os.Stdout, toDTv2Channels(channels), verbose, completed, showFailed) - return nil - }, -} - -func toDTv2Channels(channels []bapi.DataTransferChannel) []api.DataTransferChannel { - v2chs := make([]api.DataTransferChannel, 0, len(channels)) - for _, ch := range channels { - v2chs = append(v2chs, api.DataTransferChannel{ - TransferID: datatransferv2.TransferID(ch.TransferID), - Status: datatransferv2.Status(ch.Status), - BaseCID: ch.BaseCID, - IsInitiator: ch.IsInitiator, - IsSender: ch.IsSender, - Voucher: ch.Voucher, - Message: ch.Message, - OtherPeer: ch.OtherPeer, - Transferred: ch.Transferred, - Stages: toDTv2Stages(ch.Stages), - }) - } - return v2chs -} - -func toDTv2Stages(stages *datatransfer.ChannelStages) *datatransferv2.ChannelStages { - if stages == nil { - return nil - } - - v2stgs := make([]*datatransferv2.ChannelStage, 0, len(stages.Stages)) - for _, s := range stages.Stages { - v2stgs = append(v2stgs, &datatransferv2.ChannelStage{ - Name: s.Name, - Description: s.Description, - CreatedTime: s.CreatedTime, - UpdatedTime: s.UpdatedTime, - Logs: toDTv2Logs(s.Logs), - }) - } - return &datatransferv2.ChannelStages{Stages: v2stgs} -} - -func toDTv2Logs(logs []*datatransfer.Log) []*datatransferv2.Log { - if logs == nil { - return nil - } - - v2logs := make([]*datatransferv2.Log, 0, len(logs)) - for _, l := range logs { - v2logs = append(v2logs, &datatransferv2.Log{Log: l.Log, UpdatedTime: l.UpdatedTime}) - } - - return v2logs -} - -var transfersDiagnosticsCmd = &cli.Command{ - Name: "diagnostics", - Usage: "Get detailed diagnostics on active transfers with a specific peer", - ArgsUsage: "", - Flags: []cli.Flag{}, - Action: func(cctx *cli.Context) error { - if !cctx.Args().Present() { - return cli.ShowCommandHelp(cctx, cctx.Command.Name) - } - api, closer, err := lcli.GetMarketsAPI(cctx) - if err != nil { - return err - } - defer closer() - ctx := lcli.ReqContext(cctx) - - targetPeer, err := peer.Decode(cctx.Args().First()) - if err != nil { - return err - } - diagnostics, err := api.MarketDataTransferDiagnostics(ctx, targetPeer) - if err != nil { - return err - } - out, err := json.MarshalIndent(diagnostics, "", "\t") - if err != nil { - return err - } - fmt.Println(string(out)) - return nil - }, -} diff --git a/cmd/boostd/legacy_retrieval_deals.go b/cmd/boostd/legacy_retrieval_deals.go deleted file mode 100644 index 7dca34ae2..000000000 --- a/cmd/boostd/legacy_retrieval_deals.go +++ /dev/null @@ -1,277 +0,0 @@ -package main - -import ( - "fmt" - "os" - "sort" - "text/tabwriter" - - "github.com/docker/go-units" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/go-state-types/abi" - "github.com/urfave/cli/v2" - - bcli "github.com/filecoin-project/boost/cli" - "github.com/filecoin-project/lotus/chain/types" - lcli "github.com/filecoin-project/lotus/cli" -) - -var retrievalDealsCmd = &cli.Command{ - Name: "retrieval-deals", - Usage: "Manage legacy retrieval deals and related configuration (Markets V1)", - Category: "legacy", - Subcommands: []*cli.Command{ - retrievalDealSelectionCmd, - retrievalDealsListCmd, - retrievalSetAskCmd, - retrievalGetAskCmd, - }, -} - -var retrievalDealSelectionCmd = &cli.Command{ - Name: "selection", - Usage: "Configure acceptance criteria for retrieval deal proposals", - Subcommands: []*cli.Command{ - retrievalDealSelectionShowCmd, - retrievalDealSelectionResetCmd, - retrievalDealSelectionRejectCmd, - }, -} - -var retrievalDealSelectionShowCmd = &cli.Command{ - Name: "list", - Usage: "List retrieval deal proposal selection criteria", - Action: func(cctx *cli.Context) error { - smapi, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - onlineOk, err := smapi.DealsConsiderOnlineRetrievalDeals(lcli.DaemonContext(cctx)) - if err != nil { - return err - } - - offlineOk, err := smapi.DealsConsiderOfflineRetrievalDeals(lcli.DaemonContext(cctx)) - if err != nil { - return err - } - - fmt.Printf("considering online retrieval deals: %t\n", onlineOk) - fmt.Printf("considering offline retrieval deals: %t\n", offlineOk) - - return nil - }, -} - -var retrievalDealSelectionResetCmd = &cli.Command{ - Name: "reset", - Usage: "Reset retrieval deal proposal selection criteria to default values", - Action: func(cctx *cli.Context) error { - smapi, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - err = smapi.DealsSetConsiderOnlineRetrievalDeals(lcli.DaemonContext(cctx), true) - if err != nil { - return err - } - - err = smapi.DealsSetConsiderOfflineRetrievalDeals(lcli.DaemonContext(cctx), true) - if err != nil { - return err - } - - return nil - }, -} - -var retrievalDealSelectionRejectCmd = &cli.Command{ - Name: "reject", - Usage: "Configure criteria which necessitate automatic rejection", - Flags: []cli.Flag{ - &cli.BoolFlag{ - Name: "online", - }, - &cli.BoolFlag{ - Name: "offline", - }, - }, - Action: func(cctx *cli.Context) error { - smapi, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - if cctx.Bool("online") { - err = smapi.DealsSetConsiderOnlineRetrievalDeals(lcli.DaemonContext(cctx), false) - if err != nil { - return err - } - } - - if cctx.Bool("offline") { - err = smapi.DealsSetConsiderOfflineRetrievalDeals(lcli.DaemonContext(cctx), false) - if err != nil { - return err - } - } - - return nil - }, -} - -var retrievalDealsListCmd = &cli.Command{ - Name: "list", - Usage: "List all active retrieval deals for this miner", - Action: func(cctx *cli.Context) error { - api, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - deals, err := api.MarketListRetrievalDeals(lcli.DaemonContext(cctx)) - if err != nil { - return err - } - - sort.Slice(deals, func(i, j int) bool { - return deals[i].ID < deals[j].ID - }) - - w := tabwriter.NewWriter(os.Stdout, 2, 4, 2, ' ', 0) - - _, _ = fmt.Fprintf(w, "Receiver\tDealID\tPayload\tState\tPricePerByte\tBytesSent\tMessage\n") - - for _, deal := range deals { - payloadCid := deal.PayloadCID.String() - - _, _ = fmt.Fprintf(w, - "%s\t%d\t%s\t%s\t%s\t%d\t%s\n", - deal.Receiver.String(), - deal.ID, - "..."+payloadCid[len(payloadCid)-8:], - retrievalmarket.DealStatuses[deal.Status], - deal.PricePerByte.String(), - deal.TotalSent, - deal.Message, - ) - } - - return w.Flush() - }, -} - -var retrievalSetAskCmd = &cli.Command{ - Name: "set-ask", - Usage: "Configure the provider's retrieval ask", - Flags: []cli.Flag{ - &cli.StringFlag{ - Name: "price", - Usage: "Set the price of the ask for retrievals (FIL/GiB)", - }, - &cli.StringFlag{ - Name: "unseal-price", - Usage: "Set the price to unseal", - }, - &cli.StringFlag{ - Name: "payment-interval", - Usage: "Set the payment interval (in bytes) for retrieval", - DefaultText: "1MiB", - }, - &cli.StringFlag{ - Name: "payment-interval-increase", - Usage: "Set the payment interval increase (in bytes) for retrieval", - DefaultText: "1MiB", - }, - }, - Action: func(cctx *cli.Context) error { - ctx := lcli.ReqContext(cctx) - - api, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - ask, err := api.MarketGetRetrievalAsk(ctx) - if err != nil { - return err - } - - if cctx.IsSet("price") { - v, err := types.ParseFIL(cctx.String("price")) - if err != nil { - return err - } - ask.PricePerByte = types.BigDiv(types.BigInt(v), types.NewInt(1<<30)) - } - - if cctx.IsSet("unseal-price") { - v, err := types.ParseFIL(cctx.String("unseal-price")) - if err != nil { - return err - } - ask.UnsealPrice = abi.TokenAmount(v) - } - - if cctx.IsSet("payment-interval") { - v, err := units.RAMInBytes(cctx.String("payment-interval")) - if err != nil { - return err - } - ask.PaymentInterval = uint64(v) - } - - if cctx.IsSet("payment-interval-increase") { - v, err := units.RAMInBytes(cctx.String("payment-interval-increase")) - if err != nil { - return err - } - ask.PaymentIntervalIncrease = uint64(v) - } - - return api.MarketSetRetrievalAsk(ctx, ask) - }, -} - -var retrievalGetAskCmd = &cli.Command{ - Name: "get-ask", - Usage: "Get the provider's current retrieval ask configured by the provider in the ask-store using the set-ask CLI command", - Flags: []cli.Flag{}, - Action: func(cctx *cli.Context) error { - ctx := lcli.ReqContext(cctx) - - api, closer, err := bcli.GetBoostAPI(cctx) - if err != nil { - return err - } - defer closer() - - ask, err := api.MarketGetRetrievalAsk(ctx) - if err != nil { - return err - } - - w := tabwriter.NewWriter(os.Stdout, 2, 4, 2, ' ', 0) - _, _ = fmt.Fprintf(w, "Price per Byte\tUnseal Price\tPayment Interval\tPayment Interval Increase\n") - if ask == nil { - _, _ = fmt.Fprintf(w, "\n") - return w.Flush() - } - - _, _ = fmt.Fprintf(w, "%s\t%s\t%s\t%s\n", - types.FIL(ask.PricePerByte), - types.FIL(ask.UnsealPrice), - units.BytesSize(float64(ask.PaymentInterval)), - units.BytesSize(float64(ask.PaymentIntervalIncrease)), - ) - return w.Flush() - - }, -} diff --git a/cmd/boostd/main.go b/cmd/boostd/main.go index 126264e09..ac9e51594 100644 --- a/cmd/boostd/main.go +++ b/cmd/boostd/main.go @@ -36,22 +36,16 @@ func main() { authCmd, runCmd, initCmd, - migrateMonolithCmd, - migrateMarketsCmd, backupCmd, restoreCmd, configCmd, dummydealCmd, - dataTransfersCmd, - retrievalDealsCmd, indexProvCmd, importDataCmd, //importDirectDataCmd, logCmd, - dagstoreCmd, netCmd, pieceDirCmd, - recoverCmd, }, } app.Setup() diff --git a/cmd/boostd/piecedir.go b/cmd/boostd/piecedir.go index 54436b0aa..08854b6aa 100644 --- a/cmd/boostd/piecedir.go +++ b/cmd/boostd/piecedir.go @@ -16,6 +16,7 @@ var pieceDirCmd = &cli.Command{ Usage: "Manage Local Index Directory", Subcommands: []*cli.Command{ pdIndexGenerate, + recoverCmd, }, } diff --git a/cmd/boostd/recover.go b/cmd/boostd/recover.go index e21edadfb..edb20ab67 100644 --- a/cmd/boostd/recover.go +++ b/cmd/boostd/recover.go @@ -13,11 +13,11 @@ import ( "time" "github.com/davecgh/go-spew/spew" - "github.com/filecoin-project/boost-gfm/piecestore" "github.com/filecoin-project/boost/cmd/lib" "github.com/filecoin-project/boost/db" bdclient "github.com/filecoin-project/boost/extern/boostd-data/client" "github.com/filecoin-project/boost/extern/boostd-data/model" + "github.com/filecoin-project/boost/markets/piecestore" "github.com/filecoin-project/boost/node/config" "github.com/filecoin-project/boost/piecedirectory" "github.com/filecoin-project/dagstore/mount" diff --git a/cmd/booster-http/e2e_test.go b/cmd/booster-http/e2e_test.go index 24f2076ec..34094df1b 100644 --- a/cmd/booster-http/e2e_test.go +++ b/cmd/booster-http/e2e_test.go @@ -40,7 +40,7 @@ func TestE2E(t *testing.T) { framework.SetLogLevel() t.Log("Starting boost and miner") - boostAndMiner := framework.NewTestFramework(ctx, t, framework.EnableLegacyDeals(true), framework.SetMaxStagingBytes(10485760)) + boostAndMiner := framework.NewTestFramework(ctx, t, framework.SetMaxStagingBytes(10485760)) req.NoError(boostAndMiner.Start()) defer boostAndMiner.Stop() diff --git a/cmd/booster-http/piecehandler.go b/cmd/booster-http/piecehandler.go index 005c20551..f71fe040a 100644 --- a/cmd/booster-http/piecehandler.go +++ b/cmd/booster-http/piecehandler.go @@ -11,10 +11,10 @@ import ( "time" "github.com/NYTimes/gziphandler" - "github.com/filecoin-project/boost-gfm/retrievalmarket" "github.com/filecoin-project/boost/extern/boostd-data/model" "github.com/filecoin-project/boost/extern/boostd-data/shared/tracing" "github.com/filecoin-project/boost/metrics" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" "github.com/hashicorp/go-multierror" "github.com/ipfs/go-cid" "github.com/ipfs/go-datastore" @@ -143,7 +143,7 @@ func isNotFoundError(err error) bool { switch { case errors.Is(err, ErrNotFound), errors.Is(err, datastore.ErrNotFound), - errors.Is(err, retrievalmarket.ErrNotFound), + errors.Is(err, legacyretrievaltypes.ErrNotFound), strings.Contains(strings.ToLower(err.Error()), "not found"): return true default: diff --git a/cmd/booster-http/trustless_gateway_test.go b/cmd/booster-http/trustless_gateway_test.go index 49864f286..2d4bea622 100644 --- a/cmd/booster-http/trustless_gateway_test.go +++ b/cmd/booster-http/trustless_gateway_test.go @@ -31,7 +31,7 @@ func TestTrustlessGateway(t *testing.T) { kit.QuietMiningLogs() framework.SetLogLevel() - boostAndMiner := framework.NewTestFramework(ctx, t, framework.EnableLegacyDeals(true), framework.SetMaxStagingBytes(10485760)) + boostAndMiner := framework.NewTestFramework(ctx, t, framework.SetMaxStagingBytes(10485760)) req.NoError(boostAndMiner.Start()) defer boostAndMiner.Stop() diff --git a/cmd/boostx/utils_cmd.go b/cmd/boostx/utils_cmd.go index 4f34c30f8..9d798afc0 100644 --- a/cmd/boostx/utils_cmd.go +++ b/cmd/boostx/utils_cmd.go @@ -8,10 +8,10 @@ import ( "path/filepath" "time" - "github.com/filecoin-project/boost-gfm/stores" clinode "github.com/filecoin-project/boost/cli/node" "github.com/filecoin-project/boost/cmd" "github.com/filecoin-project/boost/cmd/lib" + "github.com/filecoin-project/boost/cmd/lib/stores" "github.com/filecoin-project/boost/node/config" "github.com/filecoin-project/boost/node/repo" "github.com/filecoin-project/boost/testutil" diff --git a/cmd/lib/common.go b/cmd/lib/common.go index c3bab39cc..f020fd885 100644 --- a/cmd/lib/common.go +++ b/cmd/lib/common.go @@ -9,10 +9,10 @@ import ( "os" "github.com/chzyer/readline" - "github.com/filecoin-project/boost-gfm/piecestore" - piecestoreimpl "github.com/filecoin-project/boost-gfm/piecestore/impl" - "github.com/filecoin-project/boost-gfm/storagemarket" clinode "github.com/filecoin-project/boost/cli/node" + "github.com/filecoin-project/boost/markets/piecestore" + piecestoreimpl "github.com/filecoin-project/boost/markets/piecestore/impl" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" vfsm "github.com/filecoin-project/go-ds-versioning/pkg/fsm" "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/go-state-types/big" @@ -72,7 +72,7 @@ func GetPropCidByChainDealID(ctx context.Context, ds *backupds.Datastore) (map[a } // Build a mapping of chain deal ID to proposal CID - var list []storagemarket.MinerDeal + var list []legacytypes.MinerDeal if err := deals.List(&list); err != nil { return nil, err } @@ -121,7 +121,7 @@ func getLegacyDealsFSM(ctx context.Context, ds *backupds.Datastore) (fsm.Group, // Get the deals FSM provDS := namespace.Wrap(ds, datastore.NewKey("/deals/provider")) deals, migrate, err := vfsm.NewVersionedFSM(provDS, fsm.Parameters{ - StateType: storagemarket.MinerDeal{}, + StateType: legacytypes.MinerDeal{}, StateKeyField: "State", }, nil, "2") if err != nil { diff --git a/cmd/lib/stores/dagstore.go b/cmd/lib/stores/dagstore.go new file mode 100644 index 000000000..24c722ea1 --- /dev/null +++ b/cmd/lib/stores/dagstore.go @@ -0,0 +1,12 @@ +package stores + +import ( + "io" + + bstore "github.com/ipfs/boxo/blockstore" +) + +type ClosableBlockstore interface { + bstore.Blockstore + io.Closer +} diff --git a/cmd/lib/stores/error.go b/cmd/lib/stores/error.go new file mode 100644 index 000000000..c7aecc738 --- /dev/null +++ b/cmd/lib/stores/error.go @@ -0,0 +1,11 @@ +package stores + +import ( + "errors" +) + +var ErrNotFound = errors.New("not found") + +func IsNotFound(err error) bool { + return errors.Is(err, ErrNotFound) +} diff --git a/cmd/lib/stores/filestore.go b/cmd/lib/stores/filestore.go new file mode 100644 index 000000000..31a324741 --- /dev/null +++ b/cmd/lib/stores/filestore.go @@ -0,0 +1,164 @@ +package stores + +import ( + "context" + + bstore "github.com/ipfs/boxo/blockstore" + "github.com/ipfs/boxo/filestore" + blocks "github.com/ipfs/go-block-format" + "github.com/ipfs/go-cid" + "github.com/ipfs/go-datastore" + "github.com/ipfs/go-datastore/query" + carv2 "github.com/ipld/go-car/v2" + "github.com/ipld/go-car/v2/blockstore" + mh "github.com/multiformats/go-multihash" + "golang.org/x/xerrors" +) + +// ReadOnlyFilestore opens the CAR in the specified path as as a read-only +// blockstore, and fronts it with a Filestore whose positional mappings are +// stored inside the CAR itself. It must be closed after done. +func ReadOnlyFilestore(path string) (ClosableBlockstore, error) { + ro, err := OpenReadOnly(path, + carv2.ZeroLengthSectionAsEOF(true), + blockstore.UseWholeCIDs(true), + ) + + if err != nil { + return nil, err + } + + bs, err := FilestoreOf(ro) + if err != nil { + return nil, err + } + + return &closableBlockstore{Blockstore: bs, closeFn: ro.Close}, nil +} + +// ReadWriteFilestore opens the CAR in the specified path as as a read-write +// blockstore, and fronts it with a Filestore whose positional mappings are +// stored inside the CAR itself. It must be closed after done. Closing will +// finalize the CAR blockstore. +func ReadWriteFilestore(path string, roots ...cid.Cid) (ClosableBlockstore, error) { + rw, err := OpenReadWrite(path, roots, + carv2.ZeroLengthSectionAsEOF(true), + carv2.StoreIdentityCIDs(true), + blockstore.UseWholeCIDs(true), + ) + if err != nil { + return nil, err + } + + bs, err := FilestoreOf(rw) + if err != nil { + return nil, err + } + + return &closableBlockstore{Blockstore: bs, closeFn: rw.Finalize}, nil +} + +// FilestoreOf returns a FileManager/Filestore backed entirely by a +// blockstore without requiring a datastore. It achieves this by coercing the +// blockstore into a datastore. The resulting blockstore is suitable for usage +// with DagBuilderHelper with DagBuilderParams#NoCopy=true. +func FilestoreOf(bs bstore.Blockstore) (bstore.Blockstore, error) { + coercer := &dsCoercer{bs} + + // the FileManager stores positional infos (positional mappings) in a + // datastore, which in our case is the blockstore coerced into a datastore. + // + // Passing the root dir as a base path makes me uneasy, but these filestores + // are only used locally. + fm := filestore.NewFileManager(coercer, "/") + fm.AllowFiles = true + + // the Filestore sifts leaves (PosInfos) from intermediate nodes. It writes + // PosInfo leaves to the datastore (which in our case is the coerced + // blockstore), and the intermediate nodes to the blockstore proper (since + // they cannot be mapped to the file. + fstore := filestore.NewFilestore(bs, fm) + bs = bstore.NewIdStore(fstore) + + return bs, nil +} + +var cidBuilder = cid.V1Builder{Codec: cid.Raw, MhType: mh.SHA2_256} + +// dsCoercer coerces a Blockstore to present a datastore interface, apt for +// usage with the Filestore/FileManager. Only PosInfos will be written through +// this path. +type dsCoercer struct { + bstore.Blockstore +} + +var _ datastore.Batching = (*dsCoercer)(nil) + +func (crcr *dsCoercer) Get(ctx context.Context, key datastore.Key) (value []byte, err error) { + c, err := cidBuilder.Sum(key.Bytes()) + if err != nil { + return nil, xerrors.Errorf("failed to create cid: %w", err) + } + + blk, err := crcr.Blockstore.Get(ctx, c) + if err != nil { + return nil, xerrors.Errorf("failed to get cid %s: %w", c, err) + } + return blk.RawData(), nil +} + +func (crcr *dsCoercer) Put(ctx context.Context, key datastore.Key, value []byte) error { + c, err := cidBuilder.Sum(key.Bytes()) + if err != nil { + return xerrors.Errorf("failed to create cid: %w", err) + } + blk, err := blocks.NewBlockWithCid(value, c) + if err != nil { + return xerrors.Errorf("failed to create block: %w", err) + } + if err := crcr.Blockstore.Put(ctx, blk); err != nil { + return xerrors.Errorf("failed to put block: %w", err) + } + return nil +} + +func (crcr *dsCoercer) Has(ctx context.Context, key datastore.Key) (exists bool, err error) { + c, err := cidBuilder.Sum(key.Bytes()) + if err != nil { + return false, xerrors.Errorf("failed to create cid: %w", err) + } + return crcr.Blockstore.Has(ctx, c) +} + +func (crcr *dsCoercer) Batch(_ context.Context) (datastore.Batch, error) { + return datastore.NewBasicBatch(crcr), nil +} + +func (crcr *dsCoercer) GetSize(_ context.Context, _ datastore.Key) (size int, err error) { + return 0, xerrors.New("operation NOT supported: GetSize") +} + +func (crcr *dsCoercer) Query(_ context.Context, _ query.Query) (query.Results, error) { + return nil, xerrors.New("operation NOT supported: Query") +} + +func (crcr *dsCoercer) Delete(_ context.Context, _ datastore.Key) error { + return xerrors.New("operation NOT supported: Delete") +} + +func (crcr *dsCoercer) Sync(_ context.Context, _ datastore.Key) error { + return xerrors.New("operation NOT supported: Sync") +} + +func (crcr *dsCoercer) Close() error { + return nil +} + +type closableBlockstore struct { + bstore.Blockstore + closeFn func() error +} + +func (c *closableBlockstore) Close() error { + return c.closeFn() +} diff --git a/cmd/lib/stores/kvcarbs.go b/cmd/lib/stores/kvcarbs.go new file mode 100644 index 000000000..a4cb240f2 --- /dev/null +++ b/cmd/lib/stores/kvcarbs.go @@ -0,0 +1,1676 @@ +package stores + +import ( + "bytes" + "context" + "encoding/binary" + "errors" + "fmt" + "io" + "os" + "sync" + + "github.com/ipfs/boxo/blockstore" + "github.com/ipfs/boxo/ipld/merkledag" + blocks "github.com/ipfs/go-block-format" + "github.com/ipfs/go-cid" + cbor "github.com/ipfs/go-ipld-cbor" + format "github.com/ipfs/go-ipld-format" + "github.com/ipld/go-car/util" + carv2 "github.com/ipld/go-car/v2" + "github.com/ipld/go-car/v2/index" + "github.com/multiformats/go-multicodec" + "github.com/multiformats/go-multihash" + "github.com/multiformats/go-varint" + "github.com/petar/GoLLRB/llrb" + cborg "github.com/whyrusleeping/cbor/go" + "golang.org/x/exp/mmap" +) + +/* + + This file contains extracted parts of CARv2 blockstore, modified to allow + storage of arbitrary data indexed by ID CIDs. + + This was allowed by go-car prior to v2.1.0, but newer go-car releases + require that data matches the multihash, which means that the library can + no longer be exploited as a KV store as is done in filestore.go. + + We duplicate the code here temporarily, as an alternative to breaking + existing nodes, or adding an option to go-car which would break the CAR spec + (it also contains this hack to a single repo). + + Ideally we should migrate to a real KV store, but even for that we'll still + need this code for the migration process. + +*/ + +// Modified vs go-car/v2 +func isIdentity(cid.Cid) (digest []byte, ok bool, err error) { + /* + dmh, err := multihash.Decode(key.Hash()) + if err != nil { + return nil, false, err + } + ok = dmh.Code == multihash.IDENTITY + digest = dmh.Digest + return digest, ok, nil + */ + + // This is the hack filestore datastore needs to use CARs as a KV store + return nil, false, err +} + +// Code below was copied from go-car/v2 + +var ( + _ io.ReaderAt = (*OffsetReadSeeker)(nil) + _ io.ReadSeeker = (*OffsetReadSeeker)(nil) +) + +// OffsetReadSeeker implements Read, and ReadAt on a section +// of an underlying io.ReaderAt. +// The main difference between io.SectionReader and OffsetReadSeeker is that +// NewOffsetReadSeeker does not require the user to know the number of readable bytes. +// +// It also partially implements Seek, where the implementation panics if io.SeekEnd is passed. +// This is because, OffsetReadSeeker does not know the end of the file therefore cannot seek relative +// to it. +type OffsetReadSeeker struct { + r io.ReaderAt + base int64 + off int64 +} + +// NewOffsetReadSeeker returns an OffsetReadSeeker that reads from r +// starting offset offset off and stops with io.EOF when r reaches its end. +// The Seek function will panic if whence io.SeekEnd is passed. +func NewOffsetReadSeeker(r io.ReaderAt, off int64) *OffsetReadSeeker { + return &OffsetReadSeeker{r, off, off} +} + +func (o *OffsetReadSeeker) Read(p []byte) (n int, err error) { + n, err = o.r.ReadAt(p, o.off) + o.off += int64(n) + return +} + +func (o *OffsetReadSeeker) ReadAt(p []byte, off int64) (n int, err error) { + if off < 0 { + return 0, io.EOF + } + off += o.base + return o.r.ReadAt(p, off) +} + +func (o *OffsetReadSeeker) ReadByte() (byte, error) { + b := []byte{0} + _, err := o.Read(b) + return b[0], err +} + +func (o *OffsetReadSeeker) Offset() int64 { + return o.off +} + +func (o *OffsetReadSeeker) Seek(offset int64, whence int) (int64, error) { + switch whence { + case io.SeekStart: + o.off = offset + o.base + case io.SeekCurrent: + o.off += offset + case io.SeekEnd: + panic("unsupported whence: SeekEnd") + } + return o.Position(), nil +} + +// Position returns the current position of this reader relative to the initial offset. +func (o *OffsetReadSeeker) Position() int64 { + return o.off - o.base +} + +var ( + _ io.Writer = (*OffsetWriteSeeker)(nil) + _ io.WriteSeeker = (*OffsetWriteSeeker)(nil) +) + +type OffsetWriteSeeker struct { + w io.WriterAt + base int64 + offset int64 +} + +func NewOffsetWriter(w io.WriterAt, off int64) *OffsetWriteSeeker { + return &OffsetWriteSeeker{w, off, off} +} + +func (ow *OffsetWriteSeeker) Write(b []byte) (n int, err error) { + n, err = ow.w.WriteAt(b, ow.offset) + ow.offset += int64(n) + return +} + +func (ow *OffsetWriteSeeker) Seek(offset int64, whence int) (int64, error) { + switch whence { + case io.SeekStart: + ow.offset = offset + ow.base + case io.SeekCurrent: + ow.offset += offset + case io.SeekEnd: + panic("unsupported whence: SeekEnd") + } + return ow.Position(), nil +} + +// Position returns the current position of this writer relative to the initial offset, i.e. the number of bytes written. +func (ow *OffsetWriteSeeker) Position() int64 { + return ow.offset - ow.base +} + +type BytesReader interface { + io.Reader + io.ByteReader +} + +func ReadNode(r io.Reader, zeroLenAsEOF bool) (cid.Cid, []byte, error) { + data, err := LdRead(r, zeroLenAsEOF) + if err != nil { + return cid.Cid{}, nil, err + } + + n, c, err := cid.CidFromBytes(data) + if err != nil { + return cid.Cid{}, nil, err + } + + return c, data[n:], nil +} + +func LdWrite(w io.Writer, d ...[]byte) error { + var sum uint64 + for _, s := range d { + sum += uint64(len(s)) + } + + buf := make([]byte, 8) + n := varint.PutUvarint(buf, sum) + _, err := w.Write(buf[:n]) + if err != nil { + return err + } + + for _, s := range d { + _, err = w.Write(s) + if err != nil { + return err + } + } + + return nil +} + +func LdSize(d ...[]byte) uint64 { + var sum uint64 + for _, s := range d { + sum += uint64(len(s)) + } + s := varint.UvarintSize(sum) + return sum + uint64(s) +} + +func LdRead(r io.Reader, zeroLenAsEOF bool) ([]byte, error) { + l, err := varint.ReadUvarint(ToByteReader(r)) + if err != nil { + // If the length of bytes read is non-zero when the error is EOF then signal an unclean EOF. + if l > 0 && err == io.EOF { + return nil, io.ErrUnexpectedEOF + } + return nil, err + } else if l == 0 && zeroLenAsEOF { + return nil, io.EOF + } + + buf := make([]byte, l) + if _, err := io.ReadFull(r, buf); err != nil { + return nil, err + } + + return buf, nil +} + +var ( + _ io.ByteReader = (*readerPlusByte)(nil) + _ io.ByteReader = (*readSeekerPlusByte)(nil) + _ io.ByteReader = (*discardingReadSeekerPlusByte)(nil) + _ io.ReadSeeker = (*discardingReadSeekerPlusByte)(nil) + _ io.ReaderAt = (*readSeekerAt)(nil) +) + +type ( + readerPlusByte struct { + io.Reader + + byteBuf [1]byte // escapes via io.Reader.Read; preallocate + } + + readSeekerPlusByte struct { + io.ReadSeeker + + byteBuf [1]byte // escapes via io.Reader.Read; preallocate + } + + discardingReadSeekerPlusByte struct { + io.Reader + offset int64 + + byteBuf [1]byte // escapes via io.Reader.Read; preallocate + } + + ByteReadSeeker interface { + io.ReadSeeker + io.ByteReader + } + + readSeekerAt struct { + rs io.ReadSeeker + mu sync.Mutex + } +) + +func ToByteReader(r io.Reader) io.ByteReader { + if br, ok := r.(io.ByteReader); ok { + return br + } + return &readerPlusByte{Reader: r} +} + +func ToByteReadSeeker(r io.Reader) ByteReadSeeker { + if brs, ok := r.(ByteReadSeeker); ok { + return brs + } + if rs, ok := r.(io.ReadSeeker); ok { + return &readSeekerPlusByte{ReadSeeker: rs} + } + return &discardingReadSeekerPlusByte{Reader: r} +} + +func ToReaderAt(rs io.ReadSeeker) io.ReaderAt { + if ra, ok := rs.(io.ReaderAt); ok { + return ra + } + return &readSeekerAt{rs: rs} +} + +func (rb *readerPlusByte) ReadByte() (byte, error) { + _, err := io.ReadFull(rb, rb.byteBuf[:]) + return rb.byteBuf[0], err +} + +func (rsb *readSeekerPlusByte) ReadByte() (byte, error) { + _, err := io.ReadFull(rsb, rsb.byteBuf[:]) + return rsb.byteBuf[0], err +} + +func (drsb *discardingReadSeekerPlusByte) ReadByte() (byte, error) { + _, err := io.ReadFull(drsb, drsb.byteBuf[:]) + return drsb.byteBuf[0], err +} + +func (drsb *discardingReadSeekerPlusByte) Read(p []byte) (read int, err error) { + read, err = drsb.Reader.Read(p) + drsb.offset += int64(read) + return +} + +func (drsb *discardingReadSeekerPlusByte) Seek(offset int64, whence int) (int64, error) { + switch whence { + case io.SeekStart: + n := offset - drsb.offset + if n < 0 { + panic("unsupported rewind via whence: io.SeekStart") + } + _, err := io.CopyN(io.Discard, drsb, n) + return drsb.offset, err + case io.SeekCurrent: + _, err := io.CopyN(io.Discard, drsb, offset) + return drsb.offset, err + default: + panic("unsupported whence: io.SeekEnd") + } +} + +func (rsa *readSeekerAt) ReadAt(p []byte, off int64) (n int, err error) { + rsa.mu.Lock() + defer rsa.mu.Unlock() + if _, err := rsa.rs.Seek(off, io.SeekStart); err != nil { + return 0, err + } + return rsa.rs.Read(p) +} + +func init() { + cbor.RegisterCborType(CarHeader{}) +} + +type Store interface { + Put(blocks.Block) error +} + +type ReadStore interface { + Get(cid.Cid) (blocks.Block, error) +} + +type CarHeader struct { + Roots []cid.Cid + Version uint64 +} + +type carWriter struct { + ds format.NodeGetter + w io.Writer +} + +func WriteCar(ctx context.Context, ds format.NodeGetter, roots []cid.Cid, w io.Writer) error { + h := &CarHeader{ + Roots: roots, + Version: 1, + } + + if err := WriteHeader(h, w); err != nil { + return fmt.Errorf("failed to write car header: %s", err) + } + + cw := &carWriter{ds: ds, w: w} + seen := cid.NewSet() + for _, r := range roots { + if err := merkledag.Walk(ctx, cw.enumGetLinks, r, seen.Visit); err != nil { + return err + } + } + return nil +} + +func ReadHeader(r io.Reader) (*CarHeader, error) { + hb, err := LdRead(r, false) + if err != nil { + return nil, err + } + + var ch CarHeader + if err := cbor.DecodeInto(hb, &ch); err != nil { + return nil, fmt.Errorf("invalid header: %v", err) + } + + return &ch, nil +} + +func WriteHeader(h *CarHeader, w io.Writer) error { + hb, err := cbor.DumpObject(h) + if err != nil { + return err + } + + return util.LdWrite(w, hb) +} + +func HeaderSize(h *CarHeader) (uint64, error) { + hb, err := cbor.DumpObject(h) + if err != nil { + return 0, err + } + + return util.LdSize(hb), nil +} + +func (cw *carWriter) enumGetLinks(ctx context.Context, c cid.Cid) ([]*format.Link, error) { + nd, err := cw.ds.Get(ctx, c) + if err != nil { + return nil, err + } + + if err := cw.writeNode(ctx, nd); err != nil { + return nil, err + } + + return nd.Links(), nil +} + +func (cw *carWriter) writeNode(ctx context.Context, nd format.Node) error { + return util.LdWrite(cw.w, nd.Cid().Bytes(), nd.RawData()) +} + +type CarReader struct { + r io.Reader + Header *CarHeader + zeroLenAsEOF bool +} + +func NewCarReaderWithZeroLengthSectionAsEOF(r io.Reader) (*CarReader, error) { + return newCarReader(r, true) +} + +func NewCarReader(r io.Reader) (*CarReader, error) { + return newCarReader(r, false) +} + +func newCarReader(r io.Reader, zeroLenAsEOF bool) (*CarReader, error) { + ch, err := ReadHeader(r) + if err != nil { + return nil, err + } + + if ch.Version != 1 { + return nil, fmt.Errorf("invalid car version: %d", ch.Version) + } + + if len(ch.Roots) == 0 { + return nil, fmt.Errorf("empty car, no roots") + } + + return &CarReader{ + r: r, + Header: ch, + zeroLenAsEOF: zeroLenAsEOF, + }, nil +} + +func (cr *CarReader) Next() (blocks.Block, error) { + c, data, err := ReadNode(cr.r, cr.zeroLenAsEOF) + if err != nil { + return nil, err + } + + hashed, err := c.Prefix().Sum(data) + if err != nil { + return nil, err + } + + if !hashed.Equals(c) { + return nil, fmt.Errorf("mismatch in content integrity, name: %s, data: %s", c, hashed) + } + + return blocks.NewBlockWithCid(data, c) +} + +type batchStore interface { + PutMany([]blocks.Block) error +} + +func LoadCar(s Store, r io.Reader) (*CarHeader, error) { + cr, err := NewCarReader(r) + if err != nil { + return nil, err + } + + if bs, ok := s.(batchStore); ok { + return loadCarFast(bs, cr) + } + + return loadCarSlow(s, cr) +} + +func loadCarFast(s batchStore, cr *CarReader) (*CarHeader, error) { + var buf []blocks.Block + for { + blk, err := cr.Next() + if err != nil { + if err == io.EOF { + if len(buf) > 0 { + if err := s.PutMany(buf); err != nil { + return nil, err + } + } + return cr.Header, nil + } + return nil, err + } + + buf = append(buf, blk) + + if len(buf) > 1000 { + if err := s.PutMany(buf); err != nil { + return nil, err + } + buf = buf[:0] + } + } +} + +func loadCarSlow(s Store, cr *CarReader) (*CarHeader, error) { + for { + blk, err := cr.Next() + if err != nil { + if err == io.EOF { + return cr.Header, nil + } + return nil, err + } + + if err := s.Put(blk); err != nil { + return nil, err + } + } +} + +// Matches checks whether two headers match. +// Two headers are considered matching if: +// 1. They have the same version number, and +// 2. They contain the same root CIDs in any order. +// +// Note, this function explicitly ignores the order of roots. +// If order of roots matter use reflect.DeepEqual instead. +func (h CarHeader) Matches(other CarHeader) bool { + if h.Version != other.Version { + return false + } + thisLen := len(h.Roots) + if thisLen != len(other.Roots) { + return false + } + // Headers with a single root are popular. + // Implement a fast execution path for popular cases. + if thisLen == 1 { + return h.Roots[0].Equals(other.Roots[0]) + } + + // Check other contains all roots. + // TODO: should this be optimised for cases where the number of roots are large since it has O(N^2) complexity? + for _, r := range h.Roots { + if !other.containsRoot(r) { + return false + } + } + return true +} + +func (h *CarHeader) containsRoot(root cid.Cid) bool { + for _, r := range h.Roots { + if r.Equals(root) { + return true + } + } + return false +} + +var _ blockstore.Blockstore = (*ReadOnly)(nil) + +var ( + errZeroLengthSection = fmt.Errorf("zero-length carv2 section not allowed by default; see WithZeroLengthSectionAsEOF option") + errReadOnly = fmt.Errorf("called write method on a read-only carv2 blockstore") + errClosed = fmt.Errorf("cannot use a carv2 blockstore after closing") +) + +// ReadOnly provides a read-only CAR Block Store. +type ReadOnly struct { + // mu allows ReadWrite to be safe for concurrent use. + // It's in ReadOnly so that read operations also grab read locks, + // given that ReadWrite embeds ReadOnly for methods like Get and Has. + // + // The main fields guarded by the mutex are the index and the underlying writers. + // For simplicity, the entirety of the blockstore methods grab the mutex. + mu sync.RWMutex + + // When true, the blockstore has been closed via Close, Discard, or + // Finalize, and must not be used. Any further blockstore method calls + // will return errClosed to avoid panics or broken behavior. + closed bool + + // The backing containing the data payload in CARv1 format. + backing io.ReaderAt + // The CARv1 content index. + idx index.Index + + // If we called carv2.NewReaderMmap, remember to close it too. + carv2Closer io.Closer + + opts carv2.Options +} + +type contextKey string + +const asyncErrHandlerKey contextKey = "asyncErrorHandlerKey" + +// UseWholeCIDs is a read option which makes a CAR blockstore identify blocks by +// whole CIDs, and not just their multihashes. The default is to use +// multihashes, which matches the current semantics of go-ipfs-blockstore v1. +// +// Enabling this option affects a number of methods, including read-only ones: +// +// • Get, Has, and HasSize will only return a block +// only if the entire CID is present in the CAR file. +// +// • AllKeysChan will return the original whole CIDs, instead of with their +// multicodec set to "raw" to just provide multihashes. +// +// • If AllowDuplicatePuts isn't set, +// Put and PutMany will deduplicate by the whole CID, +// allowing different CIDs with equal multihashes. +// +// Note that this option only affects the blockstore, and is ignored by the root +// go-car/v2 package. +func UseWholeCIDs(enable bool) carv2.Option { + return func(o *carv2.Options) { + o.BlockstoreUseWholeCIDs = enable + } +} + +// NewReadOnly creates a new ReadOnly blockstore from the backing with a optional index as idx. +// This function accepts both CARv1 and CARv2 backing. +// The blockstore is instantiated with the given index if it is not nil. +// +// Otherwise: +// * For a CARv1 backing an index is generated. +// * For a CARv2 backing an index is only generated if Header.HasIndex returns false. +// +// There is no need to call ReadOnly.Close on instances returned by this function. +func NewReadOnly(backing io.ReaderAt, idx index.Index, opts ...carv2.Option) (*ReadOnly, error) { + b := &ReadOnly{ + opts: carv2.ApplyOptions(opts...), + } + + version, err := readVersion(backing) + if err != nil { + return nil, err + } + switch version { + case 1: + if idx == nil { + if idx, err = generateIndex(backing, opts...); err != nil { + return nil, err + } + } + b.backing = backing + b.idx = idx + return b, nil + case 2: + v2r, err := carv2.NewReader(backing, opts...) + if err != nil { + return nil, err + } + if idx == nil { + if v2r.Header.HasIndex() { + r, err := v2r.IndexReader() + if err != nil { + return nil, err + } + idx, err = index.ReadFrom(r) + if err != nil { + return nil, err + } + } else { + r, err := v2r.DataReader() + if err != nil { + return nil, err + } + idx, err = generateIndex(r, opts...) + if err != nil { + return nil, err + } + } + } + drBacking, err := v2r.DataReader() + if err != nil { + return nil, err + } + b.backing = drBacking + b.idx = idx + return b, nil + default: + return nil, fmt.Errorf("unsupported car version: %v", version) + } +} + +func readVersion(at io.ReaderAt) (uint64, error) { + var rr io.Reader + switch r := at.(type) { + case io.Reader: + rr = r + default: + rr = NewOffsetReadSeeker(r, 0) + } + return carv2.ReadVersion(rr) +} + +func generateIndex(at io.ReaderAt, opts ...carv2.Option) (index.Index, error) { + var rs io.ReadSeeker + switch r := at.(type) { + case io.ReadSeeker: + rs = r + default: + rs = NewOffsetReadSeeker(r, 0) + } + + // Note, we do not set any write options so that all write options fall back onto defaults. + return carv2.GenerateIndex(rs, opts...) +} + +// OpenReadOnly opens a read-only blockstore from a CAR file (either v1 or v2), generating an index if it does not exist. +// Note, the generated index if the index does not exist is ephemeral and only stored in memory. +// See car.GenerateIndex and Index.Attach for persisting index onto a CAR file. +func OpenReadOnly(path string, opts ...carv2.Option) (*ReadOnly, error) { + f, err := mmap.Open(path) + if err != nil { + return nil, err + } + + robs, err := NewReadOnly(f, nil, opts...) + if err != nil { + return nil, err + } + robs.carv2Closer = f + + return robs, nil +} + +func (b *ReadOnly) readBlock(idx int64) (cid.Cid, []byte, error) { + bcid, data, err := ReadNode(NewOffsetReadSeeker(b.backing, idx), b.opts.ZeroLengthSectionAsEOF) + return bcid, data, err +} + +// DeleteBlock is unsupported and always errors. +func (b *ReadOnly) DeleteBlock(_ context.Context, _ cid.Cid) error { + return errReadOnly +} + +// Has indicates if the store contains a block that corresponds to the given key. +// This function always returns true for any given key with multihash.IDENTITY code. +func (b *ReadOnly) Has(ctx context.Context, key cid.Cid) (bool, error) { + // Check if the given CID has multihash.IDENTITY code + // Note, we do this without locking, since there is no shared information to lock for in order to perform the check. + if _, ok, err := isIdentity(key); err != nil { + return false, err + } else if ok { + return true, nil + } + + b.mu.RLock() + defer b.mu.RUnlock() + + if b.closed { + return false, errClosed + } + + var fnFound bool + var fnErr error + err := b.idx.GetAll(key, func(offset uint64) bool { + uar := NewOffsetReadSeeker(b.backing, int64(offset)) + var err error + _, err = varint.ReadUvarint(uar) + if err != nil { + fnErr = err + return false + } + _, readCid, err := cid.CidFromReader(uar) + if err != nil { + fnErr = err + return false + } + if b.opts.BlockstoreUseWholeCIDs { + fnFound = readCid.Equals(key) + return !fnFound // continue looking if we haven't found it + } else { + fnFound = bytes.Equal(readCid.Hash(), key.Hash()) + return false + } + }) + if errors.Is(err, index.ErrNotFound) { + return false, nil + } else if err != nil { + return false, err + } + return fnFound, fnErr +} + +// Get gets a block corresponding to the given key. +// This API will always return true if the given key has multihash.IDENTITY code. +func (b *ReadOnly) Get(ctx context.Context, key cid.Cid) (blocks.Block, error) { + // Check if the given CID has multihash.IDENTITY code + // Note, we do this without locking, since there is no shared information to lock for in order to perform the check. + if digest, ok, err := isIdentity(key); err != nil { + return nil, err + } else if ok { + return blocks.NewBlockWithCid(digest, key) + } + + b.mu.RLock() + defer b.mu.RUnlock() + + if b.closed { + return nil, errClosed + } + + var fnData []byte + var fnErr error + err := b.idx.GetAll(key, func(offset uint64) bool { + readCid, data, err := b.readBlock(int64(offset)) + if err != nil { + fnErr = err + return false + } + if b.opts.BlockstoreUseWholeCIDs { + if readCid.Equals(key) { + fnData = data + return false + } else { + return true // continue looking + } + } else { + if bytes.Equal(readCid.Hash(), key.Hash()) { + fnData = data + } + return false + } + }) + if errors.Is(err, index.ErrNotFound) { + return nil, format.ErrNotFound{Cid: key} + } else if err != nil { + return nil, err + } else if fnErr != nil { + return nil, fnErr + } + if fnData == nil { + return nil, format.ErrNotFound{Cid: key} + } + return blocks.NewBlockWithCid(fnData, key) +} + +// GetSize gets the size of an item corresponding to the given key. +func (b *ReadOnly) GetSize(ctx context.Context, key cid.Cid) (int, error) { + // Check if the given CID has multihash.IDENTITY code + // Note, we do this without locking, since there is no shared information to lock for in order to perform the check. + if digest, ok, err := isIdentity(key); err != nil { + return 0, err + } else if ok { + return len(digest), nil + } + + b.mu.RLock() + defer b.mu.RUnlock() + + if b.closed { + return 0, errClosed + } + + fnSize := -1 + var fnErr error + err := b.idx.GetAll(key, func(offset uint64) bool { + rdr := NewOffsetReadSeeker(b.backing, int64(offset)) + sectionLen, err := varint.ReadUvarint(rdr) + if err != nil { + fnErr = err + return false + } + cidLen, readCid, err := cid.CidFromReader(rdr) + if err != nil { + fnErr = err + return false + } + if b.opts.BlockstoreUseWholeCIDs { + if readCid.Equals(key) { + fnSize = int(sectionLen) - cidLen + return false + } else { + return true // continue looking + } + } else { + if bytes.Equal(readCid.Hash(), key.Hash()) { + fnSize = int(sectionLen) - cidLen + } + return false + } + }) + if errors.Is(err, index.ErrNotFound) { + return -1, format.ErrNotFound{Cid: key} + } else if err != nil { + return -1, err + } else if fnErr != nil { + return -1, fnErr + } + if fnSize == -1 { + return -1, format.ErrNotFound{Cid: key} + } + return fnSize, nil +} + +// Put is not supported and always returns an error. +func (b *ReadOnly) Put(context.Context, blocks.Block) error { + return errReadOnly +} + +// PutMany is not supported and always returns an error. +func (b *ReadOnly) PutMany(context.Context, []blocks.Block) error { + return errReadOnly +} + +// WithAsyncErrorHandler returns a context with async error handling set to the given errHandler. +// Any errors that occur during asynchronous operations of AllKeysChan will be passed to the given +// handler. +func WithAsyncErrorHandler(ctx context.Context, errHandler func(error)) context.Context { + return context.WithValue(ctx, asyncErrHandlerKey, errHandler) +} + +// AllKeysChan returns the list of keys in the CAR data payload. +// If the ctx is constructed using WithAsyncErrorHandler any errors that occur during asynchronous +// retrieval of CIDs will be passed to the error handler function set in context. +// Otherwise, errors will terminate the asynchronous operation silently. +// +// See WithAsyncErrorHandler +func (b *ReadOnly) AllKeysChan(ctx context.Context) (<-chan cid.Cid, error) { + // We release the lock when the channel-sending goroutine stops. + // Note that we can't use a deferred unlock here, + // because if we return a nil error, + // we only want to unlock once the async goroutine has stopped. + b.mu.RLock() + + if b.closed { + b.mu.RUnlock() // don't hold the mutex forever + return nil, errClosed + } + + // TODO we may use this walk for populating the index, and we need to be able to iterate keys in this way somewhere for index generation. In general though, when it's asked for all keys from a blockstore with an index, we should iterate through the index when possible rather than linear reads through the full car. + rdr := NewOffsetReadSeeker(b.backing, 0) + header, err := ReadHeader(rdr) + if err != nil { + b.mu.RUnlock() // don't hold the mutex forever + return nil, fmt.Errorf("error reading car header: %w", err) + } + headerSize, err := HeaderSize(header) + if err != nil { + b.mu.RUnlock() // don't hold the mutex forever + return nil, err + } + + // TODO: document this choice of 5, or use simpler buffering like 0 or 1. + ch := make(chan cid.Cid, 5) + + // Seek to the end of header. + if _, err = rdr.Seek(int64(headerSize), io.SeekStart); err != nil { + b.mu.RUnlock() // don't hold the mutex forever + return nil, err + } + + go func() { + defer b.mu.RUnlock() + defer close(ch) + + for { + length, err := varint.ReadUvarint(rdr) + if err != nil { + if err != io.EOF { + maybeReportError(ctx, err) + } + return + } + + // Null padding; by default it's an error. + if length == 0 { + if b.opts.ZeroLengthSectionAsEOF { + break + } else { + maybeReportError(ctx, errZeroLengthSection) + return + } + } + + thisItemForNxt := rdr.Offset() + _, c, err := cid.CidFromReader(rdr) + if err != nil { + maybeReportError(ctx, err) + return + } + if _, err := rdr.Seek(thisItemForNxt+int64(length), io.SeekStart); err != nil { + maybeReportError(ctx, err) + return + } + + // If we're just using multihashes, flatten to the "raw" codec. + if !b.opts.BlockstoreUseWholeCIDs { + c = cid.NewCidV1(cid.Raw, c.Hash()) + } + + select { + case ch <- c: + case <-ctx.Done(): + maybeReportError(ctx, ctx.Err()) + return + } + } + }() + return ch, nil +} + +// maybeReportError checks if an error handler is present in context associated to the key +// asyncErrHandlerKey, and if preset it will pass the error to it. +func maybeReportError(ctx context.Context, err error) { + value := ctx.Value(asyncErrHandlerKey) + if eh, _ := value.(func(error)); eh != nil { + eh(err) + } +} + +// HashOnRead is currently unimplemented; hashing on reads never happens. +func (b *ReadOnly) HashOnRead(bool) { + // TODO: implement before the final release? +} + +// Roots returns the root CIDs of the backing CAR. +func (b *ReadOnly) Roots() ([]cid.Cid, error) { + header, err := ReadHeader(NewOffsetReadSeeker(b.backing, 0)) + if err != nil { + return nil, fmt.Errorf("error reading car header: %w", err) + } + return header.Roots, nil +} + +// Close closes the underlying reader if it was opened by OpenReadOnly. +// After this call, the blockstore can no longer be used. +// +// Note that this call may block if any blockstore operations are currently in +// progress, including an AllKeysChan that hasn't been fully consumed or cancelled. +func (b *ReadOnly) Close() error { + b.mu.Lock() + defer b.mu.Unlock() + + return b.closeWithoutMutex() +} + +func (b *ReadOnly) closeWithoutMutex() error { + b.closed = true + if b.carv2Closer != nil { + return b.carv2Closer.Close() + } + return nil +} + +var ( + errUnsupported = errors.New("not supported") + insertionIndexCodec = multicodec.Code(0x300003) +) + +type ( + insertionIndex struct { + items llrb.LLRB + } + + recordDigest struct { + digest []byte + index.Record + } +) + +func (r recordDigest) Less(than llrb.Item) bool { + other, ok := than.(recordDigest) + if !ok { + return false + } + return bytes.Compare(r.digest, other.digest) < 0 +} + +func newRecordDigest(r index.Record) recordDigest { + d, err := multihash.Decode(r.Hash()) + if err != nil { + panic(err) + } + + return recordDigest{d.Digest, r} +} + +func newRecordFromCid(c cid.Cid, at uint64) recordDigest { + d, err := multihash.Decode(c.Hash()) + if err != nil { + panic(err) + } + + return recordDigest{d.Digest, index.Record{Cid: c, Offset: at}} +} + +func (ii *insertionIndex) insertNoReplace(key cid.Cid, n uint64) { + ii.items.InsertNoReplace(newRecordFromCid(key, n)) +} + +func (ii *insertionIndex) Get(c cid.Cid) (uint64, error) { + d, err := multihash.Decode(c.Hash()) + if err != nil { + return 0, err + } + entry := recordDigest{digest: d.Digest} + e := ii.items.Get(entry) + if e == nil { + return 0, index.ErrNotFound + } + r, ok := e.(recordDigest) + if !ok { + return 0, errUnsupported + } + + return r.Record.Offset, nil +} + +func (ii *insertionIndex) GetAll(c cid.Cid, fn func(uint64) bool) error { + d, err := multihash.Decode(c.Hash()) + if err != nil { + return err + } + entry := recordDigest{digest: d.Digest} + + any := false + iter := func(i llrb.Item) bool { + existing := i.(recordDigest) + if !bytes.Equal(existing.digest, entry.digest) { + // We've already looked at all entries with matching digests. + return false + } + any = true + return fn(existing.Record.Offset) + } + ii.items.AscendGreaterOrEqual(entry, iter) + if !any { + return index.ErrNotFound + } + return nil +} + +func (ii *insertionIndex) Marshal(w io.Writer) (uint64, error) { + l := uint64(0) + if err := binary.Write(w, binary.LittleEndian, int64(ii.items.Len())); err != nil { + return l, err + } + + l += 8 + var err error + iter := func(i llrb.Item) bool { + if err = cborg.Encode(w, i.(recordDigest).Record); err != nil { + return false + } + return true + } + ii.items.AscendGreaterOrEqual(ii.items.Min(), iter) + return l, err +} + +func (ii *insertionIndex) Unmarshal(r io.Reader) error { + var length int64 + if err := binary.Read(r, binary.LittleEndian, &length); err != nil { + return err + } + d := cborg.NewDecoder(r) + for i := int64(0); i < length; i++ { + var rec index.Record + if err := d.Decode(&rec); err != nil { + return err + } + ii.items.InsertNoReplace(newRecordDigest(rec)) + } + return nil +} + +func (ii *insertionIndex) Codec() multicodec.Code { + return insertionIndexCodec +} + +func (ii *insertionIndex) Load(rs []index.Record) error { + for _, r := range rs { + rec := newRecordDigest(r) + if rec.digest == nil { + return fmt.Errorf("invalid entry: %v", r) + } + ii.items.InsertNoReplace(rec) + } + return nil +} + +func newInsertionIndex() *insertionIndex { + return &insertionIndex{} +} + +// flatten returns a formatted index in the given codec for more efficient subsequent loading. +func (ii *insertionIndex) flatten(codec multicodec.Code) (index.Index, error) { + si, err := index.New(codec) + if err != nil { + return nil, err + } + rcrds := make([]index.Record, ii.items.Len()) + + idx := 0 + iter := func(i llrb.Item) bool { + rcrds[idx] = i.(recordDigest).Record + idx++ + return true + } + ii.items.AscendGreaterOrEqual(ii.items.Min(), iter) + + if err := si.Load(rcrds); err != nil { + return nil, err + } + return si, nil +} + +// note that hasExactCID is very similar to GetAll, +// but it's separate as it allows us to compare Record.Cid directly, +// whereas GetAll just provides Record.Offset. + +func (ii *insertionIndex) hasExactCID(c cid.Cid) bool { + d, err := multihash.Decode(c.Hash()) + if err != nil { + panic(err) + } + entry := recordDigest{digest: d.Digest} + + found := false + iter := func(i llrb.Item) bool { + existing := i.(recordDigest) + if !bytes.Equal(existing.digest, entry.digest) { + // We've already looked at all entries with matching digests. + return false + } + if existing.Record.Cid == c { + // We found an exact match. + found = true + return false + } + // Continue looking in ascending order. + return true + } + ii.items.AscendGreaterOrEqual(entry, iter) + return found +} + +var _ blockstore.Blockstore = (*ReadWrite)(nil) + +// ReadWrite implements a blockstore that stores blocks in CARv2 format. +// Blocks put into the blockstore can be read back once they are successfully written. +// This implementation is preferable for a write-heavy workload. +// The blocks are written immediately on Put and PutAll calls, while the index is stored in memory +// and updated incrementally. +// +// The Finalize function must be called once the putting blocks are finished. +// Upon calling Finalize header is finalized and index is written out. +// Once finalized, all read and write calls to this blockstore will result in errors. +type ReadWrite struct { + ronly ReadOnly + + f *os.File + dataWriter *OffsetWriteSeeker + idx *insertionIndex + header carv2.Header + + opts carv2.Options +} + +// AllowDuplicatePuts is a write option which makes a CAR blockstore not +// deduplicate blocks in Put and PutMany. The default is to deduplicate, +// which matches the current semantics of go-ipfs-blockstore v1. +// +// Note that this option only affects the blockstore, and is ignored by the root +// go-car/v2 package. +func AllowDuplicatePuts(allow bool) carv2.Option { + return func(o *carv2.Options) { + o.BlockstoreAllowDuplicatePuts = allow + } +} + +// OpenReadWrite creates a new ReadWrite at the given path with a provided set of root CIDs and options. +// +// ReadWrite.Finalize must be called once putting and reading blocks are no longer needed. +// Upon calling ReadWrite.Finalize the CARv2 header and index are written out onto the file and the +// backing file is closed. Once finalized, all read and write calls to this blockstore will result +// in errors. Note, ReadWrite.Finalize must be called on an open instance regardless of whether any +// blocks were put or not. +// +// If a file at given path does not exist, the instantiation will write car.Pragma and data payload +// header (i.e. the inner CARv1 header) onto the file before returning. +// +// When the given path already exists, the blockstore will attempt to resume from it. +// On resumption the existing data sections in file are re-indexed, allowing the caller to continue +// putting any remaining blocks without having to re-ingest blocks for which previous ReadWrite.Put +// returned successfully. +// +// Resumption only works on files that were created by a previous instance of a ReadWrite +// blockstore. This means a file created as a result of a successful call to OpenReadWrite can be +// resumed from as long as write operations such as ReadWrite.Put, ReadWrite.PutMany returned +// successfully. On resumption the roots argument and WithDataPadding option must match the +// previous instantiation of ReadWrite blockstore that created the file. More explicitly, the file +// resuming from must: +// 1. start with a complete CARv2 car.Pragma. +// 2. contain a complete CARv1 data header with root CIDs matching the CIDs passed to the +// constructor, starting at offset optionally padded by WithDataPadding, followed by zero or +// more complete data sections. If any corrupt data sections are present the resumption will fail. +// Note, if set previously, the blockstore must use the same WithDataPadding option as before, +// since this option is used to locate the CARv1 data payload. +// +// Note, resumption should be used with WithCidDeduplication, so that blocks that are successfully +// written into the file are not re-written. Unless, the user explicitly wants duplicate blocks. +// +// Resuming from finalized files is allowed. However, resumption will regenerate the index +// regardless by scanning every existing block in file. +func OpenReadWrite(path string, roots []cid.Cid, opts ...carv2.Option) (*ReadWrite, error) { + f, err := os.OpenFile(path, os.O_RDWR|os.O_CREATE, 0o666) // TODO: Should the user be able to configure FileMode permissions? + if err != nil { + return nil, fmt.Errorf("could not open read/write file: %w", err) + } + stat, err := f.Stat() + if err != nil { + // Note, we should not get a an os.ErrNotExist here because the flags used to open file includes os.O_CREATE + return nil, err + } + // Try and resume by default if the file size is non-zero. + resume := stat.Size() != 0 + // If construction of blockstore fails, make sure to close off the open file. + defer func() { + if err != nil { + f.Close() + } + }() + + // Instantiate block store. + // Set the header fileld before applying options since padding options may modify header. + rwbs := &ReadWrite{ + f: f, + idx: newInsertionIndex(), + header: carv2.NewHeader(0), + opts: carv2.ApplyOptions(opts...), + } + rwbs.ronly.opts = rwbs.opts + + if p := rwbs.opts.DataPadding; p > 0 { + rwbs.header = rwbs.header.WithDataPadding(p) + } + if p := rwbs.opts.IndexPadding; p > 0 { + rwbs.header = rwbs.header.WithIndexPadding(p) + } + + rwbs.dataWriter = NewOffsetWriter(rwbs.f, int64(rwbs.header.DataOffset)) + v1r := NewOffsetReadSeeker(rwbs.f, int64(rwbs.header.DataOffset)) + rwbs.ronly.backing = v1r + rwbs.ronly.idx = rwbs.idx + rwbs.ronly.carv2Closer = rwbs.f + + if resume { + if err = rwbs.resumeWithRoots(roots); err != nil { + return nil, err + } + } else { + if err = rwbs.initWithRoots(roots); err != nil { + return nil, err + } + } + + return rwbs, nil +} + +func (b *ReadWrite) initWithRoots(roots []cid.Cid) error { + if _, err := b.f.WriteAt(carv2.Pragma, 0); err != nil { + return err + } + return WriteHeader(&CarHeader{Roots: roots, Version: 1}, b.dataWriter) +} + +func (b *ReadWrite) resumeWithRoots(roots []cid.Cid) error { + // On resumption it is expected that the CARv2 Pragma, and the CARv1 header is successfully written. + // Otherwise we cannot resume from the file. + // Read pragma to assert if b.f is indeed a CARv2. + version, err := carv2.ReadVersion(b.f) + if err != nil { + // The file is not a valid CAR file and cannot resume from it. + // Or the write must have failed before pragma was written. + return err + } + if version != 2 { + // The file is not a CARv2 and we cannot resume from it. + return fmt.Errorf("cannot resume on CAR file with version %v", version) + } + + // Check if file was finalized by trying to read the CARv2 header. + // We check because if finalized the CARv1 reader behaviour needs to be adjusted since + // EOF will not signify end of CARv1 payload. i.e. index is most likely present. + var headerInFile carv2.Header + _, err = headerInFile.ReadFrom(NewOffsetReadSeeker(b.f, carv2.PragmaSize)) + + // If reading CARv2 header succeeded, and CARv1 offset in header is not zero then the file is + // most-likely finalized. Check padding and truncate the file to remove index. + // Otherwise, carry on reading the v1 payload at offset determined from b.header. + if err == nil && headerInFile.DataOffset != 0 { + if headerInFile.DataOffset != b.header.DataOffset { + // Assert that the padding on file matches the given WithDataPadding option. + wantPadding := headerInFile.DataOffset - carv2.PragmaSize - carv2.HeaderSize + gotPadding := b.header.DataOffset - carv2.PragmaSize - carv2.HeaderSize + return fmt.Errorf( + "cannot resume from file with mismatched CARv1 offset; "+ + "`WithDataPadding` option must match the padding on file. "+ + "Expected padding value of %v but got %v", wantPadding, gotPadding, + ) + } else if headerInFile.DataSize == 0 { + // If CARv1 size is zero, since CARv1 offset wasn't, then the CARv2 header was + // most-likely partially written. Since we write the header last in Finalize then the + // file most-likely contains the index and we cannot know where it starts, therefore + // can't resume. + return errors.New("corrupt CARv2 header; cannot resume from file") + } + } + + // Use the given CARv1 padding to instantiate the CARv1 reader on file. + v1r := NewOffsetReadSeeker(b.ronly.backing, 0) + header, err := ReadHeader(v1r) + if err != nil { + // Cannot read the CARv1 header; the file is most likely corrupt. + return fmt.Errorf("error reading car header: %w", err) + } + if !header.Matches(CarHeader{Roots: roots, Version: 1}) { + // Cannot resume if version and root does not match. + return errors.New("cannot resume on file with mismatching data header") + } + + if headerInFile.DataOffset != 0 { + // If header in file contains the size of car v1, then the index is most likely present. + // Since we will need to re-generate the index, as the one in file is flattened, truncate + // the file so that the Readonly.backing has the right set of bytes to deal with. + // This effectively means resuming from a finalized file will wipe its index even if there + // are no blocks put unless the user calls finalize. + if err := b.f.Truncate(int64(headerInFile.DataOffset + headerInFile.DataSize)); err != nil { + return err + } + } + // Now that CARv2 header is present on file, clear it to avoid incorrect size and offset in + // header in case blocksotre is closed without finalization and is resumed from. + if err := b.unfinalize(); err != nil { + return fmt.Errorf("could not un-finalize: %w", err) + } + + // TODO See how we can reduce duplicate code here. + // The code here comes from car.GenerateIndex. + // Copied because we need to populate an insertindex, not a sorted index. + // Producing a sorted index via generate, then converting it to insertindex is not possible. + // Because Index interface does not expose internal records. + // This may be done as part of https://github.com/ipld/go-car/issues/95 + + offset, err := HeaderSize(header) + if err != nil { + return err + } + sectionOffset := int64(0) + if sectionOffset, err = v1r.Seek(int64(offset), io.SeekStart); err != nil { + return err + } + + for { + // Grab the length of the section. + // Note that ReadUvarint wants a ByteReader. + length, err := varint.ReadUvarint(v1r) + if err != nil { + if err == io.EOF { + break + } + return err + } + + // Null padding; by default it's an error. + if length == 0 { + if b.ronly.opts.ZeroLengthSectionAsEOF { + break + } else { + return fmt.Errorf("carv1 null padding not allowed by default; see WithZeroLegthSectionAsEOF") + } + } + + // Grab the CID. + n, c, err := cid.CidFromReader(v1r) + if err != nil { + return err + } + b.idx.insertNoReplace(c, uint64(sectionOffset)) + + // Seek to the next section by skipping the block. + // The section length includes the CID, so subtract it. + if sectionOffset, err = v1r.Seek(int64(length)-int64(n), io.SeekCurrent); err != nil { + return err + } + } + // Seek to the end of last skipped block where the writer should resume writing. + _, err = b.dataWriter.Seek(sectionOffset, io.SeekStart) + return err +} + +func (b *ReadWrite) unfinalize() error { + _, err := new(carv2.Header).WriteTo(NewOffsetWriter(b.f, carv2.PragmaSize)) + return err +} + +// Put puts a given block to the underlying datastore +func (b *ReadWrite) Put(ctx context.Context, blk blocks.Block) error { + // PutMany already checks b.ronly.closed. + return b.PutMany(ctx, []blocks.Block{blk}) +} + +// PutMany puts a slice of blocks at the same time using batching +// capabilities of the underlying datastore whenever possible. +func (b *ReadWrite) PutMany(ctx context.Context, blks []blocks.Block) error { + b.ronly.mu.Lock() + defer b.ronly.mu.Unlock() + + if b.ronly.closed { + return errClosed + } + + for _, bl := range blks { + c := bl.Cid() + + // If StoreIdentityCIDs option is disabled then treat IDENTITY CIDs like IdStore. + if !b.opts.StoreIdentityCIDs { + // Check for IDENTITY CID. If IDENTITY, ignore and move to the next block. + if _, ok, err := isIdentity(c); err != nil { + return err + } else if ok { + continue + } + } + + // Check if its size is too big. + // If larger than maximum allowed size, return error. + // Note, we need to check this regardless of whether we have IDENTITY CID or not. + // Since multhihash codes other than IDENTITY can result in large digests. + cSize := uint64(len(c.Bytes())) + if cSize > b.opts.MaxIndexCidSize { + return &carv2.ErrCidTooLarge{MaxSize: b.opts.MaxIndexCidSize, CurrentSize: cSize} + } + + if !b.opts.BlockstoreAllowDuplicatePuts { + if b.ronly.opts.BlockstoreUseWholeCIDs && b.idx.hasExactCID(c) { + continue // deduplicated by CID + } + if !b.ronly.opts.BlockstoreUseWholeCIDs { + _, err := b.idx.Get(c) + if err == nil { + continue // deduplicated by hash + } + } + } + + n := uint64(b.dataWriter.Position()) + if err := util.LdWrite(b.dataWriter, c.Bytes(), bl.RawData()); err != nil { + return err + } + b.idx.insertNoReplace(c, n) + } + return nil +} + +// Discard closes this blockstore without finalizing its header and index. +// After this call, the blockstore can no longer be used. +// +// Note that this call may block if any blockstore operations are currently in +// progress, including an AllKeysChan that hasn't been fully consumed or cancelled. +func (b *ReadWrite) Discard() { + // Same semantics as ReadOnly.Close, including allowing duplicate calls. + // The only difference is that our method is called Discard, + // to further clarify that we're not properly finalizing and writing a + // CARv2 file. + b.ronly.Close() +} + +// Finalize finalizes this blockstore by writing the CARv2 header, along with flattened index +// for more efficient subsequent read. +// After this call, the blockstore can no longer be used. +func (b *ReadWrite) Finalize() error { + b.ronly.mu.Lock() + defer b.ronly.mu.Unlock() + + if b.ronly.closed { + // Allow duplicate Finalize calls, just like Close. + // Still error, just like ReadOnly.Close; it should be discarded. + return fmt.Errorf("called Finalize on a closed blockstore") + } + + // TODO check if add index option is set and don't write the index then set index offset to zero. + b.header = b.header.WithDataSize(uint64(b.dataWriter.Position())) + b.header.Characteristics.SetFullyIndexed(b.opts.StoreIdentityCIDs) + + // Note that we can't use b.Close here, as that tries to grab the same + // mutex we're holding here. + defer b.ronly.closeWithoutMutex() //nolint:errcheck + + // TODO if index not needed don't bother flattening it. + fi, err := b.idx.flatten(b.opts.IndexCodec) + if err != nil { + return err + } + if _, err := index.WriteTo(fi, NewOffsetWriter(b.f, int64(b.header.IndexOffset))); err != nil { + return err + } + if _, err := b.header.WriteTo(NewOffsetWriter(b.f, carv2.PragmaSize)); err != nil { + return err + } + + if err := b.ronly.closeWithoutMutex(); err != nil { + return err + } + return nil +} + +func (b *ReadWrite) AllKeysChan(ctx context.Context) (<-chan cid.Cid, error) { + return b.ronly.AllKeysChan(ctx) +} + +func (b *ReadWrite) Has(ctx context.Context, key cid.Cid) (bool, error) { + return b.ronly.Has(ctx, key) +} + +func (b *ReadWrite) Get(ctx context.Context, key cid.Cid) (blocks.Block, error) { + return b.ronly.Get(ctx, key) +} + +func (b *ReadWrite) GetSize(ctx context.Context, key cid.Cid) (int, error) { + return b.ronly.GetSize(ctx, key) +} + +func (b *ReadWrite) DeleteBlock(_ context.Context, _ cid.Cid) error { + return fmt.Errorf("ReadWrite blockstore does not support deleting blocks") +} + +func (b *ReadWrite) HashOnRead(enable bool) { + b.ronly.HashOnRead(enable) +} + +func (b *ReadWrite) Roots() ([]cid.Cid, error) { + return b.ronly.Roots() +} diff --git a/cmd/lib/stores/ro_bstores.go b/cmd/lib/stores/ro_bstores.go new file mode 100644 index 000000000..e4fcf95a0 --- /dev/null +++ b/cmd/lib/stores/ro_bstores.go @@ -0,0 +1,60 @@ +package stores + +import ( + "io" + "sync" + + bstore "github.com/ipfs/boxo/blockstore" + "golang.org/x/xerrors" +) + +// ReadOnlyBlockstores tracks open read blockstores. +type ReadOnlyBlockstores struct { + mu sync.RWMutex + stores map[string]bstore.Blockstore +} + +func NewReadOnlyBlockstores() *ReadOnlyBlockstores { + return &ReadOnlyBlockstores{ + stores: make(map[string]bstore.Blockstore), + } +} + +func (r *ReadOnlyBlockstores) Track(key string, bs bstore.Blockstore) (bool, error) { + r.mu.Lock() + defer r.mu.Unlock() + + if _, ok := r.stores[key]; ok { + return false, nil + } + + r.stores[key] = bs + return true, nil +} + +func (r *ReadOnlyBlockstores) Get(key string) (bstore.Blockstore, error) { + r.mu.RLock() + defer r.mu.RUnlock() + + if bs, ok := r.stores[key]; ok { + return bs, nil + } + + return nil, xerrors.Errorf("could not get blockstore for key %s: %w", key, ErrNotFound) +} + +func (r *ReadOnlyBlockstores) Untrack(key string) error { + r.mu.Lock() + defer r.mu.Unlock() + + if bs, ok := r.stores[key]; ok { + delete(r.stores, key) + if closer, ok := bs.(io.Closer); ok { + if err := closer.Close(); err != nil { + return xerrors.Errorf("failed to close read-only blockstore: %w", err) + } + } + } + + return nil +} diff --git a/cmd/lib/stores/rw_bstores.go b/cmd/lib/stores/rw_bstores.go new file mode 100644 index 000000000..526a16983 --- /dev/null +++ b/cmd/lib/stores/rw_bstores.go @@ -0,0 +1,63 @@ +package stores + +import ( + "sync" + + "github.com/ipfs/go-cid" + carv2 "github.com/ipld/go-car/v2" + "github.com/ipld/go-car/v2/blockstore" + "golang.org/x/xerrors" +) + +// ReadWriteBlockstores tracks open ReadWrite CAR blockstores. +type ReadWriteBlockstores struct { + mu sync.RWMutex + stores map[string]*blockstore.ReadWrite +} + +func NewReadWriteBlockstores() *ReadWriteBlockstores { + return &ReadWriteBlockstores{ + stores: make(map[string]*blockstore.ReadWrite), + } +} + +func (r *ReadWriteBlockstores) Get(key string) (*blockstore.ReadWrite, error) { + r.mu.RLock() + defer r.mu.RUnlock() + + if bs, ok := r.stores[key]; ok { + return bs, nil + } + return nil, xerrors.Errorf("could not get blockstore for key %s: %w", key, ErrNotFound) +} + +func (r *ReadWriteBlockstores) GetOrOpen(key string, path string, rootCid cid.Cid) (*blockstore.ReadWrite, error) { + r.mu.Lock() + defer r.mu.Unlock() + + if bs, ok := r.stores[key]; ok { + return bs, nil + } + + bs, err := blockstore.OpenReadWrite(path, []cid.Cid{rootCid}, blockstore.UseWholeCIDs(true), carv2.StoreIdentityCIDs(true)) + if err != nil { + return nil, xerrors.Errorf("failed to create read-write blockstore: %w", err) + } + r.stores[key] = bs + return bs, nil +} + +func (r *ReadWriteBlockstores) Untrack(key string) error { + r.mu.Lock() + defer r.mu.Unlock() + + if bs, ok := r.stores[key]; ok { + // If the blockstore has already been finalized, calling Finalize again + // will return an error. For our purposes it's simplest if Finalize is + // idempotent so we just ignore any error. + _ = bs.Finalize() + } + + delete(r.stores, key) + return nil +} diff --git a/cmd/migrate-lid/migrate_lid.go b/cmd/migrate-lid/migrate_lid.go index 445ce0356..50bb85fb7 100644 --- a/cmd/migrate-lid/migrate_lid.go +++ b/cmd/migrate-lid/migrate_lid.go @@ -12,7 +12,6 @@ import ( "sync/atomic" "time" - "github.com/filecoin-project/boost-gfm/piecestore" "github.com/filecoin-project/boost/cmd/lib" "github.com/filecoin-project/boost/db" "github.com/filecoin-project/boost/extern/boostd-data/ldb" @@ -21,8 +20,9 @@ import ( "github.com/filecoin-project/boost/extern/boostd-data/svc/types" "github.com/filecoin-project/boost/extern/boostd-data/yugabyte" "github.com/filecoin-project/boost/extern/boostd-data/yugabyte/migrations" + "github.com/filecoin-project/boost/markets/piecestore" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" "github.com/filecoin-project/go-address" - "github.com/filecoin-project/go-fil-markets/retrievalmarket" "github.com/filecoin-project/go-state-types/abi" lcli "github.com/filecoin-project/lotus/cli" "github.com/filecoin-project/lotus/node/modules" @@ -767,7 +767,7 @@ func migrateReversePiece(ctx context.Context, pieceCid cid.Cid, pieceDir StoreMi var pieceStoreDeals []piecestore.DealInfo pieceStorePieceInfo, err := ps.GetPieceInfo(pieceCid) if err != nil { - if !errors.Is(err, retrievalmarket.ErrNotFound) { + if !errors.Is(err, legacyretrievaltypes.ErrNotFound) { return 0, fmt.Errorf("getting piece info from piece store for piece %s", pieceCid) } } else { diff --git a/datatransfer/channelmonitor/channelmonitor.go b/datatransfer/channelmonitor/channelmonitor.go new file mode 100644 index 000000000..d87d4923f --- /dev/null +++ b/datatransfer/channelmonitor/channelmonitor.go @@ -0,0 +1,490 @@ +package channelmonitor + +import ( + "context" + "fmt" + "sync" + "time" + + "github.com/bep/debounce" + logging "github.com/ipfs/go-log/v2" + "github.com/libp2p/go-libp2p/core/peer" + "golang.org/x/xerrors" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/channels" +) + +var log = logging.Logger("dt-chanmon") + +type monitorAPI interface { + SubscribeToEvents(subscriber datatransfer.Subscriber) datatransfer.Unsubscribe + RestartDataTransferChannel(ctx context.Context, chid datatransfer.ChannelID) error + CloseDataTransferChannelWithError(ctx context.Context, chid datatransfer.ChannelID, cherr error) error + ConnectTo(context.Context, peer.ID) error + PeerID() peer.ID +} + +// Monitor watches the events for data transfer channels, and restarts +// a channel if there are timeouts / errors +type Monitor struct { + ctx context.Context + stop context.CancelFunc + mgr monitorAPI + cfg *Config + + lk sync.RWMutex + channels map[datatransfer.ChannelID]*monitoredChannel +} + +type Config struct { + // Max time to wait for other side to accept open channel request before attempting restart. + // Set to 0 to disable timeout. + AcceptTimeout time.Duration + // Debounce when restart is triggered by multiple errors + RestartDebounce time.Duration + // Backoff after restarting + RestartBackoff time.Duration + // Number of times to try to restart before failing + MaxConsecutiveRestarts uint32 + // Max time to wait for the responder to send a Complete message once all + // data has been sent. + // Set to 0 to disable timeout. + CompleteTimeout time.Duration + // Called when a restart completes successfully + OnRestartComplete func(id datatransfer.ChannelID) +} + +func NewMonitor(mgr monitorAPI, cfg *Config) *Monitor { + checkConfig(cfg) + ctx, cancel := context.WithCancel(context.Background()) + return &Monitor{ + ctx: ctx, + stop: cancel, + mgr: mgr, + cfg: cfg, + channels: make(map[datatransfer.ChannelID]*monitoredChannel), + } +} + +func checkConfig(cfg *Config) { + if cfg == nil { + return + } + + prefix := "data-transfer channel monitor config " + if cfg.AcceptTimeout < 0 { + panic(fmt.Sprintf(prefix+"AcceptTimeout is %s but must be >= 0", cfg.AcceptTimeout)) + } + if cfg.MaxConsecutiveRestarts == 0 { + panic(fmt.Sprintf(prefix+"MaxConsecutiveRestarts is %d but must be > 0", cfg.MaxConsecutiveRestarts)) + } + if cfg.CompleteTimeout < 0 { + panic(fmt.Sprintf(prefix+"CompleteTimeout is %s but must be >= 0", cfg.CompleteTimeout)) + } +} + +// AddPushChannel adds a push channel to the channel monitor +func (m *Monitor) AddPushChannel(chid datatransfer.ChannelID) *monitoredChannel { + return m.addChannel(chid, true) +} + +// AddPullChannel adds a pull channel to the channel monitor +func (m *Monitor) AddPullChannel(chid datatransfer.ChannelID) *monitoredChannel { + return m.addChannel(chid, false) +} + +// addChannel adds a channel to the channel monitor +func (m *Monitor) addChannel(chid datatransfer.ChannelID, isPush bool) *monitoredChannel { + if !m.enabled() { + return nil + } + + m.lk.Lock() + defer m.lk.Unlock() + + // Check if there is already a monitor for this channel + if _, ok := m.channels[chid]; ok { + tp := "push" + if !isPush { + tp = "pull" + } + log.Warnf("ignoring add %s channel %s: %s channel with that id already exists", + tp, chid, tp) + return nil + } + + mpc := newMonitoredChannel(m.ctx, m.mgr, chid, m.cfg, m.onMonitoredChannelShutdown) + m.channels[chid] = mpc + return mpc +} + +func (m *Monitor) Shutdown() { + // Cancel the context for the Monitor + m.stop() +} + +// onMonitoredChannelShutdown is called when a monitored channel shuts down +func (m *Monitor) onMonitoredChannelShutdown(chid datatransfer.ChannelID) { + m.lk.Lock() + defer m.lk.Unlock() + + delete(m.channels, chid) +} + +// enabled indicates whether the channel monitor is running +func (m *Monitor) enabled() bool { + return m.cfg != nil +} + +// monitoredChannel keeps track of events for a channel, and +// restarts the channel if there are connection issues +type monitoredChannel struct { + // The parentCtx is used when sending a close message for a channel, so + // that operation can continue even after the monitoredChannel is shutdown + parentCtx context.Context + ctx context.Context + cancel context.CancelFunc + mgr monitorAPI + chid datatransfer.ChannelID + cfg *Config + unsub datatransfer.Unsubscribe + restartChannelDebounced func(error) + onShutdown func(datatransfer.ChannelID) + shutdownLk sync.Mutex + + restartLk sync.RWMutex + restartedAt time.Time + restartQueued bool + consecutiveRestarts int +} + +func newMonitoredChannel( + parentCtx context.Context, + mgr monitorAPI, + chid datatransfer.ChannelID, + cfg *Config, + onShutdown func(datatransfer.ChannelID), +) *monitoredChannel { + ctx, cancel := context.WithCancel(context.Background()) + mpc := &monitoredChannel{ + parentCtx: parentCtx, + ctx: ctx, + cancel: cancel, + mgr: mgr, + chid: chid, + cfg: cfg, + onShutdown: onShutdown, + } + + // "debounce" calls to restart channel, ie if there are multiple calls in a + // short space of time, only send a message to restart the channel once + var lk sync.Mutex + var lastErr error + debouncer := debounce.New(cfg.RestartDebounce) + mpc.restartChannelDebounced = func(err error) { + // Log the error at debug level + log.Debug(err.Error()) + + // Save the last error passed to restartChannelDebounced + lk.Lock() + lastErr = err + lk.Unlock() + + debouncer(func() { + // Log only the last error passed to restartChannelDebounced at warning level + lk.Lock() + log.Warnf("%s", lastErr) + lk.Unlock() + + // Restart the channel + mpc.restartChannel() + }) + } + + // Start monitoring the channel + mpc.start() + return mpc +} + +// Cancel the context and unsubscribe from events. +// Returns true if channel has not already been shutdown. +func (mc *monitoredChannel) Shutdown() bool { + mc.shutdownLk.Lock() + defer mc.shutdownLk.Unlock() + + // Check if the channel was already shut down + if mc.cancel == nil { + return false + } + mc.cancel() // cancel context so all go-routines exit + mc.cancel = nil + + // unsubscribe from data transfer events + mc.unsub() + + // Inform the Manager that this channel has shut down + go mc.onShutdown(mc.chid) + + return true +} + +func (mc *monitoredChannel) start() { + // Prevent shutdown until after startup + mc.shutdownLk.Lock() + defer mc.shutdownLk.Unlock() + + log.Debugf("%s: starting data-transfer channel monitoring", mc.chid) + + // Watch to make sure the responder accepts the channel in time + cancelAcceptTimer := mc.watchForResponderAccept() + + // Watch for data-transfer channel events + mc.unsub = mc.mgr.SubscribeToEvents(func(event datatransfer.Event, channelState datatransfer.ChannelState) { + if channelState.ChannelID() != mc.chid { + return + } + + // Once the channel completes, shut down the monitor + state := channelState.Status() + if channels.IsChannelCleaningUp(state) || channels.IsChannelTerminated(state) { + log.Debugf("%s: stopping data-transfer channel monitoring (event: %s / state: %s)", + mc.chid, datatransfer.Events[event.Code], datatransfer.Statuses[channelState.Status()]) + go mc.Shutdown() + return + } + + switch event.Code { + case datatransfer.Accept: + // The Accept event is fired when we receive an Accept message from the responder + cancelAcceptTimer() + case datatransfer.SendDataError: + // If the transport layer reports an error sending data over the wire, + // attempt to restart the channel + err := xerrors.Errorf("%s: data transfer transport send error, restarting data transfer", mc.chid) + go mc.restartChannelDebounced(err) + case datatransfer.ReceiveDataError: + // If the transport layer reports an error receiving data over the wire, + // attempt to restart the channel + err := xerrors.Errorf("%s: data transfer transport receive error, restarting data transfer", mc.chid) + go mc.restartChannelDebounced(err) + case datatransfer.FinishTransfer: + // The channel initiator has finished sending / receiving all data. + // Watch to make sure that the responder sends a message to acknowledge + // that the transfer is complete + go mc.watchForResponderComplete() + case datatransfer.DataSent, datatransfer.DataReceived: + // Some data was sent / received so reset the consecutive restart + // counter + mc.resetConsecutiveRestarts() + } + }) +} + +// watchForResponderAccept watches to make sure that the responder sends +// an Accept to our open channel request before the accept timeout. +// Returns a function that can be used to cancel the timer. +func (mc *monitoredChannel) watchForResponderAccept() func() { + // Check if the accept timeout is disabled + if mc.cfg.AcceptTimeout == 0 { + return func() {} + } + + // Start a timer for the accept timeout + timer := time.NewTimer(mc.cfg.AcceptTimeout) + + go func() { + defer timer.Stop() + + select { + case <-mc.ctx.Done(): + case <-timer.C: + // Timer expired before we received an Accept from the responder, + // fail the data transfer + err := xerrors.Errorf("%s: timed out waiting %s for Accept message from remote peer", + mc.chid, mc.cfg.AcceptTimeout) + mc.closeChannelAndShutdown(err) + } + }() + + return func() { timer.Stop() } +} + +// Wait up to the configured timeout for the responder to send a Complete message +func (mc *monitoredChannel) watchForResponderComplete() { + // Check if the complete timeout is disabled + if mc.cfg.CompleteTimeout == 0 { + return + } + + // Start a timer for the complete timeout + timer := time.NewTimer(mc.cfg.CompleteTimeout) + defer timer.Stop() + + select { + case <-mc.ctx.Done(): + // When the Complete message is received, the channel shuts down and + // its context is cancelled + case <-timer.C: + // Timer expired before we received a Complete message from the responder + err := xerrors.Errorf("%s: timed out waiting %s for Complete message from remote peer", + mc.chid, mc.cfg.CompleteTimeout) + mc.closeChannelAndShutdown(err) + } +} + +// clear the consecutive restart count (we do this when data is sent or +// received) +func (mc *monitoredChannel) resetConsecutiveRestarts() { + mc.restartLk.Lock() + defer mc.restartLk.Unlock() + + mc.consecutiveRestarts = 0 +} + +// Send a restart message for the channel +func (mc *monitoredChannel) restartChannel() { + var restartedAt time.Time + mc.restartLk.Lock() + { + restartedAt = mc.restartedAt + if mc.restartedAt.IsZero() { + // If there is not already a restart in progress, we'll restart now + mc.restartedAt = time.Now() + } else { + // There is already a restart in progress, so queue up a restart + // for after the current one has completed + mc.restartQueued = true + } + } + mc.restartLk.Unlock() + + // Check if channel is already being restarted + if !restartedAt.IsZero() { + log.Debugf("%s: restart called but already restarting channel, "+ + "waiting to restart again (since %s; restart backoff is %s)", + mc.chid, time.Since(restartedAt), mc.cfg.RestartBackoff) + return + } + + for { + // Send the restart message + err := mc.doRestartChannel() + if err != nil { + // If there was an error restarting, close the channel and shutdown + // the monitor + mc.closeChannelAndShutdown(err) + return + } + + // Restart has completed, check if there's another restart queued up + restartAgain := false + mc.restartLk.Lock() + { + if mc.restartQueued { + // There's another restart queued up, so reset the restart time + // to now + mc.restartedAt = time.Now() + restartAgain = true + mc.restartQueued = false + } else { + // No other restarts queued up, so clear the restart time + mc.restartedAt = time.Time{} + } + } + mc.restartLk.Unlock() + + if !restartAgain { + // No restart queued, we're done + if mc.cfg.OnRestartComplete != nil { + mc.cfg.OnRestartComplete(mc.chid) + } + return + } + + // There was a restart queued, restart again + log.Debugf("%s: restart was queued - restarting again", mc.chid) + } +} + +func (mc *monitoredChannel) doRestartChannel() error { + // Keep track of the number of consecutive restarts with no data + // transferred + mc.restartLk.Lock() + mc.consecutiveRestarts++ + restartCount := mc.consecutiveRestarts + mc.restartLk.Unlock() + + if uint32(restartCount) > mc.cfg.MaxConsecutiveRestarts { + // If no data has been transferred since the last restart, and we've + // reached the consecutive restart limit, return an error + return xerrors.Errorf("%s: after %d consecutive restarts failed to transfer any data", mc.chid, restartCount) + } + + // Send the restart message + log.Debugf("%s: restarting (%d consecutive restarts)", mc.chid, restartCount) + err := mc.sendRestartMessage(restartCount) + if err != nil { + log.Warnf("%s: restart failed, trying again: %s", mc.chid, err) + // If the restart message could not be sent, try again + return mc.doRestartChannel() + } + log.Infof("%s: restart completed successfully", mc.chid) + + return nil +} + +func (mc *monitoredChannel) sendRestartMessage(restartCount int) error { + // Establish a connection to the peer, in case the connection went down. + // Note that at the networking layer there is logic to retry if a network + // connection cannot be established, so this may take some time. + p := mc.chid.OtherParty(mc.mgr.PeerID()) + log.Debugf("%s: re-establishing connection to %s", mc.chid, p) + start := time.Now() + err := mc.mgr.ConnectTo(mc.ctx, p) + if err != nil { + return xerrors.Errorf("%s: failed to reconnect to peer %s after %s: %w", + mc.chid, p, time.Since(start), err) + } + log.Debugf("%s: re-established connection to %s in %s", mc.chid, p, time.Since(start)) + + // Send a restart message for the channel + log.Debugf("%s: sending restart message to %s (%d consecutive restarts)", mc.chid, p, restartCount) + err = mc.mgr.RestartDataTransferChannel(mc.ctx, mc.chid) + if err != nil { + return xerrors.Errorf("%s: failed to send restart message to %s: %w", mc.chid, p, err) + } + + // The restart message was sent successfully. + // If a restart backoff is configured, backoff after a restart before + // attempting another. + if mc.cfg.RestartBackoff > 0 { + log.Debugf("%s: backing off %s before allowing any other restarts", + mc.chid, mc.cfg.RestartBackoff) + select { + case <-time.After(mc.cfg.RestartBackoff): + log.Debugf("%s: restart back-off of %s complete", mc.chid, mc.cfg.RestartBackoff) + case <-mc.ctx.Done(): + return nil + } + } + + return nil +} + +// Shut down the monitor and close the data transfer channel +func (mc *monitoredChannel) closeChannelAndShutdown(cherr error) { + // Shutdown the monitor + firstShutdown := mc.Shutdown() + if !firstShutdown { + // Channel was already shutdown, ignore this second attempt to shutdown + return + } + + // Close the data transfer channel and fire an error + log.Errorf("%s: closing data-transfer channel: %s", mc.chid, cherr) + err := mc.mgr.CloseDataTransferChannelWithError(mc.parentCtx, mc.chid, cherr) + if err != nil { + log.Errorf("error closing data-transfer channel %s: %s", mc.chid, err) + } +} diff --git a/datatransfer/channels/block_index_cache.go b/datatransfer/channels/block_index_cache.go new file mode 100644 index 000000000..42142a30f --- /dev/null +++ b/datatransfer/channels/block_index_cache.go @@ -0,0 +1,63 @@ +package channels + +import ( + "sync" + "sync/atomic" + + "github.com/filecoin-project/boost/datatransfer" +) + +type readOriginalFn func(datatransfer.ChannelID) (int64, error) + +type blockIndexKey struct { + evt datatransfer.EventCode + chid datatransfer.ChannelID +} +type blockIndexCache struct { + lk sync.RWMutex + values map[blockIndexKey]*int64 +} + +func newBlockIndexCache() *blockIndexCache { + return &blockIndexCache{ + values: make(map[blockIndexKey]*int64), + } +} + +func (bic *blockIndexCache) getValue(evt datatransfer.EventCode, chid datatransfer.ChannelID, readFromOriginal readOriginalFn) (*int64, error) { + idxKey := blockIndexKey{evt, chid} + bic.lk.RLock() + value := bic.values[idxKey] + bic.lk.RUnlock() + if value != nil { + return value, nil + } + bic.lk.Lock() + defer bic.lk.Unlock() + value = bic.values[idxKey] + if value != nil { + return value, nil + } + newValue, err := readFromOriginal(chid) + if err != nil { + return nil, err + } + bic.values[idxKey] = &newValue + return &newValue, nil +} + +func (bic *blockIndexCache) updateIfGreater(evt datatransfer.EventCode, chid datatransfer.ChannelID, newIndex int64, readFromOriginal readOriginalFn) (bool, error) { + value, err := bic.getValue(evt, chid, readFromOriginal) + if err != nil { + return false, err + } + for { + currentIndex := atomic.LoadInt64(value) + if newIndex <= currentIndex { + return false, nil + } + if atomic.CompareAndSwapInt64(value, currentIndex, newIndex) { + return true, nil + } + } +} diff --git a/datatransfer/channels/channel_state.go b/datatransfer/channels/channel_state.go new file mode 100644 index 000000000..3de3baa59 --- /dev/null +++ b/datatransfer/channels/channel_state.go @@ -0,0 +1,251 @@ +package channels + +import ( + "bytes" + + "github.com/ipfs/go-cid" + "github.com/ipld/go-ipld-prime" + "github.com/ipld/go-ipld-prime/codec/dagcbor" + basicnode "github.com/ipld/go-ipld-prime/node/basic" + "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/channels/internal" +) + +// channelState is immutable channel data plus mutable state +type channelState struct { + // peerId of the manager peer + selfPeer peer.ID + // an identifier for this channel shared by request and responder, set by requester through protocol + transferID datatransfer.TransferID + // base CID for the piece being transferred + baseCid cid.Cid + // portion of Piece to return, specified by an IPLD selector + selector *cbg.Deferred + // the party that is sending the data (not who initiated the request) + sender peer.ID + // the party that is receiving the data (not who initiated the request) + recipient peer.ID + // expected amount of data to be transferred + totalSize uint64 + // current status of this deal + status datatransfer.Status + // isPull indicates if this is a push or pull request + isPull bool + // total bytes read from this node and queued for sending (0 if receiver) + queued uint64 + // total bytes sent from this node (0 if receiver) + sent uint64 + // total bytes received by this node (0 if sender) + received uint64 + // number of blocks that have been received, including blocks that are + // present in more than one place in the DAG + receivedBlocksTotal int64 + // Number of blocks that have been queued, including blocks that are + // present in more than one place in the DAG + queuedBlocksTotal int64 + // Number of blocks that have been sent, including blocks that are + // present in more than one place in the DAG + sentBlocksTotal int64 + // more informative status on a channel + message string + // additional vouchers + vouchers []internal.EncodedVoucher + // additional voucherResults + voucherResults []internal.EncodedVoucherResult + voucherResultDecoder DecoderByTypeFunc + voucherDecoder DecoderByTypeFunc + + // stages tracks the timeline of events related to a data transfer, for + // traceability purposes. + stages *datatransfer.ChannelStages +} + +// EmptyChannelState is the zero value for channel state, meaning not present +var EmptyChannelState = channelState{} + +// Status is the current status of this channel +func (c channelState) Status() datatransfer.Status { return c.status } + +// Received returns the number of bytes received +func (c channelState) Queued() uint64 { return c.queued } + +// Sent returns the number of bytes sent +func (c channelState) Sent() uint64 { return c.sent } + +// Received returns the number of bytes received +func (c channelState) Received() uint64 { return c.received } + +// TransferID returns the transfer id for this channel +func (c channelState) TransferID() datatransfer.TransferID { return c.transferID } + +// BaseCID returns the CID that is at the root of this data transfer +func (c channelState) BaseCID() cid.Cid { return c.baseCid } + +// Selector returns the IPLD selector for this data transfer (represented as +// an IPLD node) +func (c channelState) Selector() ipld.Node { + builder := basicnode.Prototype.Any.NewBuilder() + reader := bytes.NewReader(c.selector.Raw) + err := dagcbor.Decode(builder, reader) + if err != nil { + log.Error(err) + } + return builder.Build() +} + +// Voucher returns the voucher for this data transfer +func (c channelState) Voucher() datatransfer.Voucher { + if len(c.vouchers) == 0 { + return nil + } + decoder, has := c.voucherDecoder(c.vouchers[0].Type) + if !has { + return nil + } + encodable, _ := decoder.DecodeFromCbor(c.vouchers[0].Voucher.Raw) + return encodable.(datatransfer.Voucher) +} + +// ReceivedCidsTotal returns the number of (non-unique) cids received so far +// on the channel - note that a block can exist in more than one place in the DAG +func (c channelState) ReceivedCidsTotal() int64 { + return c.receivedBlocksTotal +} + +// QueuedCidsTotal returns the number of (non-unique) cids queued so far +// on the channel - note that a block can exist in more than one place in the DAG +func (c channelState) QueuedCidsTotal() int64 { + return c.queuedBlocksTotal +} + +// SentCidsTotal returns the number of (non-unique) cids sent so far +// on the channel - note that a block can exist in more than one place in the DAG +func (c channelState) SentCidsTotal() int64 { + return c.sentBlocksTotal +} + +// Sender returns the peer id for the node that is sending data +func (c channelState) Sender() peer.ID { return c.sender } + +// Recipient returns the peer id for the node that is receiving data +func (c channelState) Recipient() peer.ID { return c.recipient } + +// TotalSize returns the total size for the data being transferred +func (c channelState) TotalSize() uint64 { return c.totalSize } + +// IsPull returns whether this is a pull request based on who initiated it +func (c channelState) IsPull() bool { + return c.isPull +} + +func (c channelState) ChannelID() datatransfer.ChannelID { + if c.isPull { + return datatransfer.ChannelID{ID: c.transferID, Initiator: c.recipient, Responder: c.sender} + } + return datatransfer.ChannelID{ID: c.transferID, Initiator: c.sender, Responder: c.recipient} +} + +func (c channelState) Message() string { + return c.message +} + +func (c channelState) Vouchers() []datatransfer.Voucher { + vouchers := make([]datatransfer.Voucher, 0, len(c.vouchers)) + for _, encoded := range c.vouchers { + decoder, has := c.voucherDecoder(encoded.Type) + if !has { + continue + } + encodable, _ := decoder.DecodeFromCbor(encoded.Voucher.Raw) + vouchers = append(vouchers, encodable.(datatransfer.Voucher)) + } + return vouchers +} + +func (c channelState) LastVoucher() datatransfer.Voucher { + decoder, has := c.voucherDecoder(c.vouchers[len(c.vouchers)-1].Type) + if !has { + return nil + } + encodable, _ := decoder.DecodeFromCbor(c.vouchers[len(c.vouchers)-1].Voucher.Raw) + return encodable.(datatransfer.Voucher) +} + +func (c channelState) LastVoucherResult() datatransfer.VoucherResult { + decoder, has := c.voucherResultDecoder(c.voucherResults[len(c.voucherResults)-1].Type) + if !has { + return nil + } + encodable, _ := decoder.DecodeFromCbor(c.voucherResults[len(c.voucherResults)-1].VoucherResult.Raw) + return encodable.(datatransfer.VoucherResult) +} + +func (c channelState) VoucherResults() []datatransfer.VoucherResult { + voucherResults := make([]datatransfer.VoucherResult, 0, len(c.voucherResults)) + for _, encoded := range c.voucherResults { + decoder, has := c.voucherResultDecoder(encoded.Type) + if !has { + continue + } + encodable, _ := decoder.DecodeFromCbor(encoded.VoucherResult.Raw) + voucherResults = append(voucherResults, encodable.(datatransfer.VoucherResult)) + } + return voucherResults +} + +func (c channelState) SelfPeer() peer.ID { + return c.selfPeer +} + +func (c channelState) OtherPeer() peer.ID { + if c.sender == c.selfPeer { + return c.recipient + } + return c.sender +} + +// Stages returns the current ChannelStages object, or an empty object. +// It is unsafe for the caller to modify the return value, and changes may not +// be persisted. It should be treated as immutable. +// +// EXPERIMENTAL; subject to change. +func (c channelState) Stages() *datatransfer.ChannelStages { + if c.stages == nil { + // return an empty placeholder; it will be discarded because the caller + // is not supposed to mutate the value anyway. + return &datatransfer.ChannelStages{} + } + + return c.stages +} + +func fromInternalChannelState(c internal.ChannelState, voucherDecoder DecoderByTypeFunc, voucherResultDecoder DecoderByTypeFunc) datatransfer.ChannelState { + return channelState{ + selfPeer: c.SelfPeer, + isPull: c.Initiator == c.Recipient, + transferID: c.TransferID, + baseCid: c.BaseCid, + selector: c.Selector, + sender: c.Sender, + recipient: c.Recipient, + totalSize: c.TotalSize, + status: c.Status, + queued: c.Queued, + sent: c.Sent, + received: c.Received, + receivedBlocksTotal: c.ReceivedBlocksTotal, + queuedBlocksTotal: c.QueuedBlocksTotal, + sentBlocksTotal: c.SentBlocksTotal, + message: c.Message, + vouchers: c.Vouchers, + voucherResults: c.VoucherResults, + voucherResultDecoder: voucherResultDecoder, + voucherDecoder: voucherDecoder, + stages: c.Stages, + } +} + +var _ datatransfer.ChannelState = channelState{} diff --git a/datatransfer/channels/channels.go b/datatransfer/channels/channels.go new file mode 100644 index 000000000..67d98332f --- /dev/null +++ b/datatransfer/channels/channels.go @@ -0,0 +1,414 @@ +package channels + +import ( + "context" + "errors" + "time" + + "github.com/ipfs/go-cid" + "github.com/ipfs/go-datastore" + "github.com/ipld/go-ipld-prime" + "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + "golang.org/x/xerrors" + + versioning "github.com/filecoin-project/go-ds-versioning/pkg" + versionedfsm "github.com/filecoin-project/go-ds-versioning/pkg/fsm" + "github.com/filecoin-project/go-statemachine" + "github.com/filecoin-project/go-statemachine/fsm" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/channels/internal" + "github.com/filecoin-project/boost/datatransfer/channels/internal/migrations" + "github.com/filecoin-project/boost/datatransfer/encoding" +) + +type DecoderByTypeFunc func(identifier datatransfer.TypeIdentifier) (encoding.Decoder, bool) + +type Notifier func(datatransfer.Event, datatransfer.ChannelState) + +// ErrNotFound is returned when a channel cannot be found with a given channel ID +type ErrNotFound struct { + ChannelID datatransfer.ChannelID +} + +func (e *ErrNotFound) Error() string { + return "No channel for channel ID " + e.ChannelID.String() +} + +func NewErrNotFound(chid datatransfer.ChannelID) error { + return &ErrNotFound{ChannelID: chid} +} + +// ErrWrongType is returned when a caller attempts to change the type of implementation data after setting it +var ErrWrongType = errors.New("Cannot change type of implementation specific data after setting it") + +// Channels is a thread safe list of channels +type Channels struct { + notifier Notifier + voucherDecoder DecoderByTypeFunc + voucherResultDecoder DecoderByTypeFunc + blockIndexCache *blockIndexCache + stateMachines fsm.Group + migrateStateMachines func(context.Context) error +} + +// ChannelEnvironment -- just a proxy for DTNetwork for now +type ChannelEnvironment interface { + Protect(id peer.ID, tag string) + Unprotect(id peer.ID, tag string) bool + ID() peer.ID + CleanupChannel(chid datatransfer.ChannelID) +} + +// New returns a new thread safe list of channels +func New(ds datastore.Batching, + notifier Notifier, + voucherDecoder DecoderByTypeFunc, + voucherResultDecoder DecoderByTypeFunc, + env ChannelEnvironment, + selfPeer peer.ID) (*Channels, error) { + + c := &Channels{ + notifier: notifier, + voucherDecoder: voucherDecoder, + voucherResultDecoder: voucherResultDecoder, + } + c.blockIndexCache = newBlockIndexCache() + channelMigrations, err := migrations.GetChannelStateMigrations(selfPeer) + if err != nil { + return nil, err + } + c.stateMachines, c.migrateStateMachines, err = versionedfsm.NewVersionedFSM(ds, fsm.Parameters{ + Environment: env, + StateType: internal.ChannelState{}, + StateKeyField: "Status", + Events: ChannelEvents, + StateEntryFuncs: ChannelStateEntryFuncs, + Notifier: c.dispatch, + FinalityStates: ChannelFinalityStates, + }, channelMigrations, versioning.VersionKey("2")) + if err != nil { + return nil, err + } + return c, nil +} + +// Start migrates the channel data store as needed +func (c *Channels) Start(ctx context.Context) error { + return c.migrateStateMachines(ctx) +} + +func (c *Channels) dispatch(eventName fsm.EventName, channel fsm.StateType) { + evtCode, ok := eventName.(datatransfer.EventCode) + if !ok { + log.Errorf("dropped bad event %v", eventName) + } + realChannel, ok := channel.(internal.ChannelState) + if !ok { + log.Errorf("not a ClientDeal %v", channel) + } + evt := datatransfer.Event{ + Code: evtCode, + Message: realChannel.Message, + Timestamp: time.Now(), + } + log.Debugw("process data transfer listeners", "name", datatransfer.Events[evtCode], "transfer ID", realChannel.TransferID) + c.notifier(evt, c.fromInternalChannelState(realChannel)) +} + +// CreateNew creates a new channel id and channel state and saves to channels. +// returns error if the channel exists already. +func (c *Channels) CreateNew(selfPeer peer.ID, tid datatransfer.TransferID, baseCid cid.Cid, selector ipld.Node, voucher datatransfer.Voucher, initiator, dataSender, dataReceiver peer.ID) (datatransfer.ChannelID, error) { + var responder peer.ID + if dataSender == initiator { + responder = dataReceiver + } else { + responder = dataSender + } + chid := datatransfer.ChannelID{Initiator: initiator, Responder: responder, ID: tid} + voucherBytes, err := encoding.Encode(voucher) + if err != nil { + return datatransfer.ChannelID{}, err + } + selBytes, err := encoding.Encode(selector) + if err != nil { + return datatransfer.ChannelID{}, err + } + err = c.stateMachines.Begin(chid, &internal.ChannelState{ + SelfPeer: selfPeer, + TransferID: tid, + Initiator: initiator, + Responder: responder, + BaseCid: baseCid, + Selector: &cbg.Deferred{Raw: selBytes}, + Sender: dataSender, + Recipient: dataReceiver, + Stages: &datatransfer.ChannelStages{}, + Vouchers: []internal.EncodedVoucher{ + { + Type: voucher.Type(), + Voucher: &cbg.Deferred{ + Raw: voucherBytes, + }, + }, + }, + Status: datatransfer.Requested, + }) + if err != nil { + log.Errorw("failed to create new tracking channel for data-transfer", "channelID", chid, "err", err) + return datatransfer.ChannelID{}, err + } + log.Debugw("created tracking channel for data-transfer, emitting channel Open event", "channelID", chid) + return chid, c.stateMachines.Send(chid, datatransfer.Open) +} + +// InProgress returns a list of in progress channels +func (c *Channels) InProgress() (map[datatransfer.ChannelID]datatransfer.ChannelState, error) { + var internalChannels []internal.ChannelState + err := c.stateMachines.List(&internalChannels) + if err != nil { + return nil, err + } + channels := make(map[datatransfer.ChannelID]datatransfer.ChannelState, len(internalChannels)) + for _, internalChannel := range internalChannels { + channels[datatransfer.ChannelID{ID: internalChannel.TransferID, Responder: internalChannel.Responder, Initiator: internalChannel.Initiator}] = + c.fromInternalChannelState(internalChannel) + } + return channels, nil +} + +// GetByID searches for a channel in the slice of channels with id `chid`. +// Returns datatransfer.EmptyChannelState if there is no channel with that id +func (c *Channels) GetByID(ctx context.Context, chid datatransfer.ChannelID) (datatransfer.ChannelState, error) { + var internalChannel internal.ChannelState + err := c.stateMachines.GetSync(ctx, chid, &internalChannel) + if err != nil { + return nil, NewErrNotFound(chid) + } + return c.fromInternalChannelState(internalChannel), nil +} + +// Accept marks a data transfer as accepted +func (c *Channels) Accept(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.Accept) +} + +func (c *Channels) ChannelOpened(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.Opened) +} + +func (c *Channels) TransferRequestQueued(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.TransferRequestQueued) +} + +// Restart marks a data transfer as restarted +func (c *Channels) Restart(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.Restart) +} + +func (c *Channels) CompleteCleanupOnRestart(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.CompleteCleanupOnRestart) +} + +func (c *Channels) getQueuedIndex(chid datatransfer.ChannelID) (int64, error) { + chst, err := c.GetByID(context.TODO(), chid) + if err != nil { + return 0, err + } + return chst.QueuedCidsTotal(), nil +} + +func (c *Channels) getReceivedIndex(chid datatransfer.ChannelID) (int64, error) { + chst, err := c.GetByID(context.TODO(), chid) + if err != nil { + return 0, err + } + return chst.ReceivedCidsTotal(), nil +} + +func (c *Channels) getSentIndex(chid datatransfer.ChannelID) (int64, error) { + chst, err := c.GetByID(context.TODO(), chid) + if err != nil { + return 0, err + } + return chst.SentCidsTotal(), nil +} + +func (c *Channels) DataSent(chid datatransfer.ChannelID, k cid.Cid, delta uint64, index int64, unique bool) (bool, error) { + return c.fireProgressEvent(chid, datatransfer.DataSent, datatransfer.DataSentProgress, k, delta, index, unique, c.getSentIndex) +} + +func (c *Channels) DataQueued(chid datatransfer.ChannelID, k cid.Cid, delta uint64, index int64, unique bool) (bool, error) { + return c.fireProgressEvent(chid, datatransfer.DataQueued, datatransfer.DataQueuedProgress, k, delta, index, unique, c.getQueuedIndex) +} + +// Returns true if this is the first time the block has been received +func (c *Channels) DataReceived(chid datatransfer.ChannelID, k cid.Cid, delta uint64, index int64, unique bool) (bool, error) { + new, err := c.fireProgressEvent(chid, datatransfer.DataReceived, datatransfer.DataReceivedProgress, k, delta, index, unique, c.getReceivedIndex) + return new, err +} + +// PauseInitiator pauses the initator of this channel +func (c *Channels) PauseInitiator(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.PauseInitiator) +} + +// PauseResponder pauses the responder of this channel +func (c *Channels) PauseResponder(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.PauseResponder) +} + +// ResumeInitiator resumes the initator of this channel +func (c *Channels) ResumeInitiator(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.ResumeInitiator) +} + +// ResumeResponder resumes the responder of this channel +func (c *Channels) ResumeResponder(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.ResumeResponder) +} + +// NewVoucher records a new voucher for this channel +func (c *Channels) NewVoucher(chid datatransfer.ChannelID, voucher datatransfer.Voucher) error { + voucherBytes, err := encoding.Encode(voucher) + if err != nil { + return err + } + return c.send(chid, datatransfer.NewVoucher, voucher.Type(), voucherBytes) +} + +// NewVoucherResult records a new voucher result for this channel +func (c *Channels) NewVoucherResult(chid datatransfer.ChannelID, voucherResult datatransfer.VoucherResult) error { + voucherResultBytes, err := encoding.Encode(voucherResult) + if err != nil { + return err + } + return c.send(chid, datatransfer.NewVoucherResult, voucherResult.Type(), voucherResultBytes) +} + +// Complete indicates responder has completed sending/receiving data +func (c *Channels) Complete(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.Complete) +} + +// FinishTransfer an initiator has finished sending/receiving data +func (c *Channels) FinishTransfer(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.FinishTransfer) +} + +// ResponderCompletes indicates an initator has finished receiving data from a responder +func (c *Channels) ResponderCompletes(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.ResponderCompletes) +} + +// ResponderBeginsFinalization indicates a responder has finished processing but is awaiting confirmation from the initiator +func (c *Channels) ResponderBeginsFinalization(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.ResponderBeginsFinalization) +} + +// BeginFinalizing indicates a responder has finished processing but is awaiting confirmation from the initiator +func (c *Channels) BeginFinalizing(chid datatransfer.ChannelID) error { + return c.send(chid, datatransfer.BeginFinalizing) +} + +// Cancel indicates a channel was cancelled prematurely +func (c *Channels) Cancel(chid datatransfer.ChannelID) error { + err := c.send(chid, datatransfer.Cancel) + + // If there was an error because the state machine already terminated, + // sending a Cancel event is redundant anyway, so just ignore it + if errors.Is(err, statemachine.ErrTerminated) { + return nil + } + + return err +} + +// Error indicates something that went wrong on a channel +func (c *Channels) Error(chid datatransfer.ChannelID, err error) error { + return c.send(chid, datatransfer.Error, err) +} + +// Disconnected indicates that the connection went down and it was not possible +// to restart it +func (c *Channels) Disconnected(chid datatransfer.ChannelID, err error) error { + return c.send(chid, datatransfer.Disconnected, err) +} + +// RequestCancelled indicates that a transport layer request was cancelled by the +// request opener +func (c *Channels) RequestCancelled(chid datatransfer.ChannelID, err error) error { + return c.send(chid, datatransfer.RequestCancelled, err) +} + +// SendDataError indicates that the transport layer had an error trying +// to send data to the remote peer +func (c *Channels) SendDataError(chid datatransfer.ChannelID, err error) error { + return c.send(chid, datatransfer.SendDataError, err) +} + +// ReceiveDataError indicates that the transport layer had an error receiving +// data from the remote peer +func (c *Channels) ReceiveDataError(chid datatransfer.ChannelID, err error) error { + return c.send(chid, datatransfer.ReceiveDataError, err) +} + +// HasChannel returns true if the given channel id is being tracked +func (c *Channels) HasChannel(chid datatransfer.ChannelID) (bool, error) { + return c.stateMachines.Has(chid) +} + +// fireProgressEvent fires +// - an event for queuing / sending / receiving blocks +// - a corresponding "progress" event if the block has not been seen before +// For example, if a block is being sent for the first time, the method will +// fire both DataSent AND DataSentProgress. +// If a block is resent, the method will fire DataSent but not DataSentProgress. +// Returns true if the block is new (both the event and a progress event were fired). +func (c *Channels) fireProgressEvent(chid datatransfer.ChannelID, evt datatransfer.EventCode, progressEvt datatransfer.EventCode, k cid.Cid, delta uint64, index int64, unique bool, readFromOriginal readOriginalFn) (bool, error) { + if err := c.checkChannelExists(chid, evt); err != nil { + return false, err + } + + isNewIndex, err := c.blockIndexCache.updateIfGreater(evt, chid, index, readFromOriginal) + if err != nil { + return false, err + } + + // If the block has not been seen before, fire the progress event + if unique && isNewIndex { + if err := c.stateMachines.Send(chid, progressEvt, delta); err != nil { + return false, err + } + } + + // Fire the regular event + return unique && isNewIndex, c.stateMachines.Send(chid, evt, index) +} + +func (c *Channels) send(chid datatransfer.ChannelID, code datatransfer.EventCode, args ...interface{}) error { + err := c.checkChannelExists(chid, code) + if err != nil { + return err + } + log.Debugw("send data transfer event", "name", datatransfer.Events[code], "transfer ID", chid.ID, "args", args) + return c.stateMachines.Send(chid, code, args...) +} + +func (c *Channels) checkChannelExists(chid datatransfer.ChannelID, code datatransfer.EventCode) error { + has, err := c.stateMachines.Has(chid) + if err != nil { + return err + } + if !has { + return xerrors.Errorf("cannot send FSM event %s to data-transfer channel %s: %w", + datatransfer.Events[code], chid, NewErrNotFound(chid)) + } + return nil +} + +// Convert from the internally used channel state format to the externally exposed ChannelState +func (c *Channels) fromInternalChannelState(ch internal.ChannelState) datatransfer.ChannelState { + return fromInternalChannelState(ch, c.voucherDecoder, c.voucherResultDecoder) +} diff --git a/datatransfer/channels/channels_fsm.go b/datatransfer/channels/channels_fsm.go new file mode 100644 index 000000000..2b8111e59 --- /dev/null +++ b/datatransfer/channels/channels_fsm.go @@ -0,0 +1,298 @@ +package channels + +import ( + logging "github.com/ipfs/go-log/v2" + cbg "github.com/whyrusleeping/cbor-gen" + + "github.com/filecoin-project/go-statemachine/fsm" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/channels/internal" +) + +var log = logging.Logger("data-transfer") + +var transferringStates = []fsm.StateKey{ + datatransfer.Requested, + datatransfer.Ongoing, + datatransfer.InitiatorPaused, + datatransfer.ResponderPaused, + datatransfer.BothPaused, + datatransfer.ResponderCompleted, + datatransfer.ResponderFinalizing, +} + +// ChannelEvents describe the events taht can +var ChannelEvents = fsm.Events{ + // Open a channel + fsm.Event(datatransfer.Open).FromAny().To(datatransfer.Requested).Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), + // Remote peer has accepted the Open channel request + fsm.Event(datatransfer.Accept).From(datatransfer.Requested).To(datatransfer.Ongoing).Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), + + fsm.Event(datatransfer.TransferRequestQueued).FromAny().ToJustRecord().Action(func(chst *internal.ChannelState) error { + chst.Message = "" + chst.AddLog("") + return nil + }), + + fsm.Event(datatransfer.Restart).FromAny().ToJustRecord().Action(func(chst *internal.ChannelState) error { + chst.Message = "" + chst.AddLog("") + return nil + }), + fsm.Event(datatransfer.Cancel).FromAny().To(datatransfer.Cancelling).Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), + + // When a channel is Opened, clear any previous error message. + // (eg if the channel is opened after being restarted due to a connection + // error) + fsm.Event(datatransfer.Opened).FromAny().ToJustRecord().Action(func(chst *internal.ChannelState) error { + chst.Message = "" + chst.AddLog("") + return nil + }), + + fsm.Event(datatransfer.DataReceived).FromAny().ToNoChange(). + Action(func(chst *internal.ChannelState, rcvdBlocksTotal int64) error { + if rcvdBlocksTotal > chst.ReceivedBlocksTotal { + chst.ReceivedBlocksTotal = rcvdBlocksTotal + } + chst.AddLog("") + return nil + }), + fsm.Event(datatransfer.DataReceivedProgress).FromMany(transferringStates...).ToNoChange(). + Action(func(chst *internal.ChannelState, delta uint64) error { + chst.Received += delta + chst.AddLog("received data") + return nil + }), + + fsm.Event(datatransfer.DataSent). + FromMany(transferringStates...).ToNoChange(). + From(datatransfer.TransferFinished).ToNoChange(). + Action(func(chst *internal.ChannelState, sentBlocksTotal int64) error { + if sentBlocksTotal > chst.SentBlocksTotal { + chst.SentBlocksTotal = sentBlocksTotal + } + chst.AddLog("") + return nil + }), + + fsm.Event(datatransfer.DataSentProgress).FromMany(transferringStates...).ToNoChange(). + Action(func(chst *internal.ChannelState, delta uint64) error { + chst.Sent += delta + chst.AddLog("sending data") + return nil + }), + + fsm.Event(datatransfer.DataQueued). + FromMany(transferringStates...).ToNoChange(). + From(datatransfer.TransferFinished).ToNoChange(). + Action(func(chst *internal.ChannelState, queuedBlocksTotal int64) error { + if queuedBlocksTotal > chst.QueuedBlocksTotal { + chst.QueuedBlocksTotal = queuedBlocksTotal + } + chst.AddLog("") + return nil + }), + fsm.Event(datatransfer.DataQueuedProgress).FromMany(transferringStates...).ToNoChange(). + Action(func(chst *internal.ChannelState, delta uint64) error { + chst.Queued += delta + chst.AddLog("") + return nil + }), + fsm.Event(datatransfer.Disconnected).FromAny().ToNoChange().Action(func(chst *internal.ChannelState, err error) error { + chst.Message = err.Error() + chst.AddLog("data transfer disconnected: %s", chst.Message) + return nil + }), + fsm.Event(datatransfer.SendDataError).FromAny().ToNoChange().Action(func(chst *internal.ChannelState, err error) error { + chst.Message = err.Error() + chst.AddLog("data transfer send error: %s", chst.Message) + return nil + }), + fsm.Event(datatransfer.ReceiveDataError).FromAny().ToNoChange().Action(func(chst *internal.ChannelState, err error) error { + chst.Message = err.Error() + chst.AddLog("data transfer receive error: %s", chst.Message) + return nil + }), + fsm.Event(datatransfer.RequestCancelled).FromAny().ToNoChange().Action(func(chst *internal.ChannelState, err error) error { + chst.Message = err.Error() + chst.AddLog("data transfer request cancelled: %s", chst.Message) + return nil + }), + fsm.Event(datatransfer.Error).FromAny().To(datatransfer.Failing).Action(func(chst *internal.ChannelState, err error) error { + chst.Message = err.Error() + chst.AddLog("data transfer erred: %s", chst.Message) + return nil + }), + + fsm.Event(datatransfer.NewVoucher).FromAny().ToNoChange(). + Action(func(chst *internal.ChannelState, vtype datatransfer.TypeIdentifier, voucherBytes []byte) error { + chst.Vouchers = append(chst.Vouchers, internal.EncodedVoucher{Type: vtype, Voucher: &cbg.Deferred{Raw: voucherBytes}}) + chst.AddLog("got new voucher") + return nil + }), + fsm.Event(datatransfer.NewVoucherResult).FromAny().ToNoChange(). + Action(func(chst *internal.ChannelState, vtype datatransfer.TypeIdentifier, voucherResultBytes []byte) error { + chst.VoucherResults = append(chst.VoucherResults, + internal.EncodedVoucherResult{Type: vtype, VoucherResult: &cbg.Deferred{Raw: voucherResultBytes}}) + chst.AddLog("got new voucher result") + return nil + }), + + fsm.Event(datatransfer.PauseInitiator). + FromMany(datatransfer.Requested, datatransfer.Ongoing).To(datatransfer.InitiatorPaused). + From(datatransfer.ResponderPaused).To(datatransfer.BothPaused). + FromAny().ToJustRecord().Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), + + fsm.Event(datatransfer.PauseResponder). + FromMany(datatransfer.Requested, datatransfer.Ongoing).To(datatransfer.ResponderPaused). + From(datatransfer.InitiatorPaused).To(datatransfer.BothPaused). + FromAny().ToJustRecord().Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), + + fsm.Event(datatransfer.ResumeInitiator). + From(datatransfer.InitiatorPaused).To(datatransfer.Ongoing). + From(datatransfer.BothPaused).To(datatransfer.ResponderPaused). + FromAny().ToJustRecord().Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), + + fsm.Event(datatransfer.ResumeResponder). + From(datatransfer.ResponderPaused).To(datatransfer.Ongoing). + From(datatransfer.BothPaused).To(datatransfer.InitiatorPaused). + From(datatransfer.Finalizing).To(datatransfer.Completing). + FromAny().ToJustRecord().Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), + + // The transfer has finished on the local node - all data was sent / received + fsm.Event(datatransfer.FinishTransfer). + FromAny().To(datatransfer.TransferFinished). + FromMany(datatransfer.Failing, datatransfer.Cancelling).ToJustRecord(). + From(datatransfer.ResponderCompleted).To(datatransfer.Completing). + From(datatransfer.ResponderFinalizing).To(datatransfer.ResponderFinalizingTransferFinished). + // If we are in the requested state, it means the other party simply never responded to our + // our data transfer, or we never actually contacted them. In any case, it's safe to skip + // the finalization process and complete the transfer + From(datatransfer.Requested).To(datatransfer.Completing). + Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), + + fsm.Event(datatransfer.ResponderBeginsFinalization). + FromAny().To(datatransfer.ResponderFinalizing). + FromMany(datatransfer.Failing, datatransfer.Cancelling).ToJustRecord(). + From(datatransfer.TransferFinished).To(datatransfer.ResponderFinalizingTransferFinished).Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), + + // The remote peer sent a Complete message, meaning it has sent / received all data + fsm.Event(datatransfer.ResponderCompletes). + FromAny().To(datatransfer.ResponderCompleted). + FromMany(datatransfer.Failing, datatransfer.Cancelling).ToJustRecord(). + From(datatransfer.ResponderPaused).To(datatransfer.ResponderFinalizing). + From(datatransfer.TransferFinished).To(datatransfer.Completing). + From(datatransfer.ResponderFinalizing).To(datatransfer.ResponderCompleted). + From(datatransfer.ResponderFinalizingTransferFinished).To(datatransfer.Completing).Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), + + fsm.Event(datatransfer.BeginFinalizing).FromAny().To(datatransfer.Finalizing).Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), + + // Both the local node and the remote peer have completed the transfer + fsm.Event(datatransfer.Complete).FromAny().To(datatransfer.Completing).Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), + + fsm.Event(datatransfer.CleanupComplete). + From(datatransfer.Cancelling).To(datatransfer.Cancelled). + From(datatransfer.Failing).To(datatransfer.Failed). + From(datatransfer.Completing).To(datatransfer.Completed).Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), + + // will kickoff state handlers for channels that were cleaning up + fsm.Event(datatransfer.CompleteCleanupOnRestart).FromAny().ToNoChange().Action(func(chst *internal.ChannelState) error { + chst.AddLog("") + return nil + }), +} + +// ChannelStateEntryFuncs are handlers called as we enter different states +// (currently unused for this fsm) +var ChannelStateEntryFuncs = fsm.StateEntryFuncs{ + datatransfer.Cancelling: cleanupConnection, + datatransfer.Failing: cleanupConnection, + datatransfer.Completing: cleanupConnection, +} + +func cleanupConnection(ctx fsm.Context, env ChannelEnvironment, channel internal.ChannelState) error { + otherParty := channel.Initiator + if otherParty == env.ID() { + otherParty = channel.Responder + } + env.CleanupChannel(datatransfer.ChannelID{ID: channel.TransferID, Initiator: channel.Initiator, Responder: channel.Responder}) + env.Unprotect(otherParty, datatransfer.ChannelID{ID: channel.TransferID, Initiator: channel.Initiator, Responder: channel.Responder}.String()) + return ctx.Trigger(datatransfer.CleanupComplete) +} + +// CleanupStates are the penultimate states for a channel +var CleanupStates = []fsm.StateKey{ + datatransfer.Cancelling, + datatransfer.Completing, + datatransfer.Failing, +} + +// ChannelFinalityStates are the final states for a channel +var ChannelFinalityStates = []fsm.StateKey{ + datatransfer.Cancelled, + datatransfer.Completed, + datatransfer.Failed, +} + +// IsChannelTerminated returns true if the channel is in a finality state +func IsChannelTerminated(st datatransfer.Status) bool { + for _, s := range ChannelFinalityStates { + if s == st { + return true + } + } + + return false +} + +// IsChannelCleaningUp returns true if channel was being cleaned up and finished +func IsChannelCleaningUp(st datatransfer.Status) bool { + for _, s := range CleanupStates { + if s == st { + return true + } + } + + return false +} diff --git a/datatransfer/channels/internal/internalchannel.go b/datatransfer/channels/internal/internalchannel.go new file mode 100644 index 000000000..f112d92a3 --- /dev/null +++ b/datatransfer/channels/internal/internalchannel.go @@ -0,0 +1,90 @@ +package internal + +import ( + "fmt" + + "github.com/ipfs/go-cid" + "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + + "github.com/filecoin-project/boost/datatransfer" +) + +//go:generate cbor-gen-for --map-encoding ChannelState EncodedVoucher EncodedVoucherResult + +// EncodedVoucher is how the voucher is stored on disk +type EncodedVoucher struct { + // Vouchers identifier for decoding + Type datatransfer.TypeIdentifier + // used to verify this channel + Voucher *cbg.Deferred +} + +// EncodedVoucherResult is how the voucher result is stored on disk +type EncodedVoucherResult struct { + // Vouchers identifier for decoding + Type datatransfer.TypeIdentifier + // used to verify this channel + VoucherResult *cbg.Deferred +} + +// ChannelState is the internal representation on disk for the channel fsm +type ChannelState struct { + // PeerId of the manager peer + SelfPeer peer.ID + // an identifier for this channel shared by request and responder, set by requester through protocol + TransferID datatransfer.TransferID + // Initiator is the person who intiated this datatransfer request + Initiator peer.ID + // Responder is the person who is responding to this datatransfer request + Responder peer.ID + // base CID for the piece being transferred + BaseCid cid.Cid + // portion of Piece to return, specified by an IPLD selector + Selector *cbg.Deferred + // the party that is sending the data (not who initiated the request) + Sender peer.ID + // the party that is receiving the data (not who initiated the request) + Recipient peer.ID + // expected amount of data to be transferred + TotalSize uint64 + // current status of this deal + Status datatransfer.Status + // total bytes read from this node and queued for sending (0 if receiver) + Queued uint64 + // total bytes sent from this node (0 if receiver) + Sent uint64 + // total bytes received by this node (0 if sender) + Received uint64 + // more informative status on a channel + Message string + Vouchers []EncodedVoucher + VoucherResults []EncodedVoucherResult + // Number of blocks that have been received, including blocks that are + // present in more than one place in the DAG + ReceivedBlocksTotal int64 + // Number of blocks that have been queued, including blocks that are + // present in more than one place in the DAG + QueuedBlocksTotal int64 + // Number of blocks that have been sent, including blocks that are + // present in more than one place in the DAG + SentBlocksTotal int64 + // Stages traces the execution fo a data transfer. + // + // EXPERIMENTAL; subject to change. + Stages *datatransfer.ChannelStages +} + +// AddLog takes an fmt string with arguments, and adds the formatted string to +// the logs for the current deal stage. +// +// EXPERIMENTAL; subject to change. +func (cs *ChannelState) AddLog(msg string, a ...interface{}) { + if len(a) > 0 { + msg = fmt.Sprintf(msg, a...) + } + + stage := datatransfer.Statuses[cs.Status] + + cs.Stages.AddLog(stage, msg) +} diff --git a/datatransfer/channels/internal/internalchannel_cbor_gen.go b/datatransfer/channels/internal/internalchannel_cbor_gen.go new file mode 100644 index 000000000..343eaa94c --- /dev/null +++ b/datatransfer/channels/internal/internalchannel_cbor_gen.go @@ -0,0 +1,1083 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package internal + +import ( + "fmt" + "io" + "math" + "sort" + + datatransfer "github.com/filecoin-project/boost/datatransfer" + cid "github.com/ipfs/go-cid" + peer "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + xerrors "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +func (t *ChannelState) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{180}); err != nil { + return err + } + + // t.Sent (uint64) (uint64) + if len("Sent") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Sent\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Sent"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Sent")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Sent)); err != nil { + return err + } + + // t.Queued (uint64) (uint64) + if len("Queued") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Queued\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Queued"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Queued")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Queued)); err != nil { + return err + } + + // t.Sender (peer.ID) (string) + if len("Sender") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Sender\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Sender"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Sender")); err != nil { + return err + } + + if len(t.Sender) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Sender was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Sender))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Sender)); err != nil { + return err + } + + // t.Stages (datatransfer.ChannelStages) (struct) + if len("Stages") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Stages\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Stages"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Stages")); err != nil { + return err + } + + if err := t.Stages.MarshalCBOR(cw); err != nil { + return err + } + + // t.Status (datatransfer.Status) (uint64) + if len("Status") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Status\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Status"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Status")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Status)); err != nil { + return err + } + + // t.BaseCid (cid.Cid) (struct) + if len("BaseCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"BaseCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("BaseCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("BaseCid")); err != nil { + return err + } + + if err := cbg.WriteCid(cw, t.BaseCid); err != nil { + return xerrors.Errorf("failed to write cid field t.BaseCid: %w", err) + } + + // t.Message (string) (string) + if len("Message") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Message\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Message"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Message")); err != nil { + return err + } + + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.Received (uint64) (uint64) + if len("Received") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Received\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Received"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Received")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Received)); err != nil { + return err + } + + // t.Selector (typegen.Deferred) (struct) + if len("Selector") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Selector\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Selector"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Selector")); err != nil { + return err + } + + if err := t.Selector.MarshalCBOR(cw); err != nil { + return err + } + + // t.SelfPeer (peer.ID) (string) + if len("SelfPeer") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"SelfPeer\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("SelfPeer"))); err != nil { + return err + } + if _, err := cw.WriteString(string("SelfPeer")); err != nil { + return err + } + + if len(t.SelfPeer) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.SelfPeer was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.SelfPeer))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.SelfPeer)); err != nil { + return err + } + + // t.Vouchers ([]internal.EncodedVoucher) (slice) + if len("Vouchers") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Vouchers\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Vouchers"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Vouchers")); err != nil { + return err + } + + if len(t.Vouchers) > cbg.MaxLength { + return xerrors.Errorf("Slice value in field t.Vouchers was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajArray, uint64(len(t.Vouchers))); err != nil { + return err + } + for _, v := range t.Vouchers { + if err := v.MarshalCBOR(cw); err != nil { + return err + } + + } + + // t.Initiator (peer.ID) (string) + if len("Initiator") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Initiator\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Initiator"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Initiator")); err != nil { + return err + } + + if len(t.Initiator) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Initiator was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Initiator))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Initiator)); err != nil { + return err + } + + // t.Recipient (peer.ID) (string) + if len("Recipient") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Recipient\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Recipient"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Recipient")); err != nil { + return err + } + + if len(t.Recipient) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Recipient was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Recipient))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Recipient)); err != nil { + return err + } + + // t.Responder (peer.ID) (string) + if len("Responder") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Responder\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Responder"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Responder")); err != nil { + return err + } + + if len(t.Responder) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Responder was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Responder))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Responder)); err != nil { + return err + } + + // t.TotalSize (uint64) (uint64) + if len("TotalSize") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"TotalSize\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("TotalSize"))); err != nil { + return err + } + if _, err := cw.WriteString(string("TotalSize")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.TotalSize)); err != nil { + return err + } + + // t.TransferID (datatransfer.TransferID) (uint64) + if len("TransferID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"TransferID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("TransferID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("TransferID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.TransferID)); err != nil { + return err + } + + // t.VoucherResults ([]internal.EncodedVoucherResult) (slice) + if len("VoucherResults") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"VoucherResults\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("VoucherResults"))); err != nil { + return err + } + if _, err := cw.WriteString(string("VoucherResults")); err != nil { + return err + } + + if len(t.VoucherResults) > cbg.MaxLength { + return xerrors.Errorf("Slice value in field t.VoucherResults was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajArray, uint64(len(t.VoucherResults))); err != nil { + return err + } + for _, v := range t.VoucherResults { + if err := v.MarshalCBOR(cw); err != nil { + return err + } + + } + + // t.SentBlocksTotal (int64) (int64) + if len("SentBlocksTotal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"SentBlocksTotal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("SentBlocksTotal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("SentBlocksTotal")); err != nil { + return err + } + + if t.SentBlocksTotal >= 0 { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.SentBlocksTotal)); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajNegativeInt, uint64(-t.SentBlocksTotal-1)); err != nil { + return err + } + } + + // t.QueuedBlocksTotal (int64) (int64) + if len("QueuedBlocksTotal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"QueuedBlocksTotal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("QueuedBlocksTotal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("QueuedBlocksTotal")); err != nil { + return err + } + + if t.QueuedBlocksTotal >= 0 { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.QueuedBlocksTotal)); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajNegativeInt, uint64(-t.QueuedBlocksTotal-1)); err != nil { + return err + } + } + + // t.ReceivedBlocksTotal (int64) (int64) + if len("ReceivedBlocksTotal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ReceivedBlocksTotal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ReceivedBlocksTotal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ReceivedBlocksTotal")); err != nil { + return err + } + + if t.ReceivedBlocksTotal >= 0 { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.ReceivedBlocksTotal)); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajNegativeInt, uint64(-t.ReceivedBlocksTotal-1)); err != nil { + return err + } + } + return nil +} + +func (t *ChannelState) UnmarshalCBOR(r io.Reader) (err error) { + *t = ChannelState{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("ChannelState: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Sent (uint64) (uint64) + case "Sent": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Sent = uint64(extra) + + } + // t.Queued (uint64) (uint64) + case "Queued": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Queued = uint64(extra) + + } + // t.Sender (peer.ID) (string) + case "Sender": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Sender = peer.ID(sval) + } + // t.Stages (datatransfer.ChannelStages) (struct) + case "Stages": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Stages = new(datatransfer.ChannelStages) + if err := t.Stages.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Stages pointer: %w", err) + } + } + + } + // t.Status (datatransfer.Status) (uint64) + case "Status": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Status = datatransfer.Status(extra) + + } + // t.BaseCid (cid.Cid) (struct) + case "BaseCid": + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.BaseCid: %w", err) + } + + t.BaseCid = c + + } + // t.Message (string) (string) + case "Message": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.Received (uint64) (uint64) + case "Received": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Received = uint64(extra) + + } + // t.Selector (typegen.Deferred) (struct) + case "Selector": + + { + + t.Selector = new(cbg.Deferred) + + if err := t.Selector.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("failed to read deferred field: %w", err) + } + } + // t.SelfPeer (peer.ID) (string) + case "SelfPeer": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.SelfPeer = peer.ID(sval) + } + // t.Vouchers ([]internal.EncodedVoucher) (slice) + case "Vouchers": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + + if extra > cbg.MaxLength { + return fmt.Errorf("t.Vouchers: array too large (%d)", extra) + } + + if maj != cbg.MajArray { + return fmt.Errorf("expected cbor array") + } + + if extra > 0 { + t.Vouchers = make([]EncodedVoucher, extra) + } + + for i := 0; i < int(extra); i++ { + { + var maj byte + var extra uint64 + var err error + _ = maj + _ = extra + _ = err + + { + + if err := t.Vouchers[i].UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Vouchers[i]: %w", err) + } + + } + + } + } + // t.Initiator (peer.ID) (string) + case "Initiator": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Initiator = peer.ID(sval) + } + // t.Recipient (peer.ID) (string) + case "Recipient": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Recipient = peer.ID(sval) + } + // t.Responder (peer.ID) (string) + case "Responder": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Responder = peer.ID(sval) + } + // t.TotalSize (uint64) (uint64) + case "TotalSize": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.TotalSize = uint64(extra) + + } + // t.TransferID (datatransfer.TransferID) (uint64) + case "TransferID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.TransferID = datatransfer.TransferID(extra) + + } + // t.VoucherResults ([]internal.EncodedVoucherResult) (slice) + case "VoucherResults": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + + if extra > cbg.MaxLength { + return fmt.Errorf("t.VoucherResults: array too large (%d)", extra) + } + + if maj != cbg.MajArray { + return fmt.Errorf("expected cbor array") + } + + if extra > 0 { + t.VoucherResults = make([]EncodedVoucherResult, extra) + } + + for i := 0; i < int(extra); i++ { + { + var maj byte + var extra uint64 + var err error + _ = maj + _ = extra + _ = err + + { + + if err := t.VoucherResults[i].UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.VoucherResults[i]: %w", err) + } + + } + + } + } + // t.SentBlocksTotal (int64) (int64) + case "SentBlocksTotal": + { + maj, extra, err := cr.ReadHeader() + var extraI int64 + if err != nil { + return err + } + switch maj { + case cbg.MajUnsignedInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 positive overflow") + } + case cbg.MajNegativeInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 negative overflow") + } + extraI = -1 - extraI + default: + return fmt.Errorf("wrong type for int64 field: %d", maj) + } + + t.SentBlocksTotal = int64(extraI) + } + // t.QueuedBlocksTotal (int64) (int64) + case "QueuedBlocksTotal": + { + maj, extra, err := cr.ReadHeader() + var extraI int64 + if err != nil { + return err + } + switch maj { + case cbg.MajUnsignedInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 positive overflow") + } + case cbg.MajNegativeInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 negative overflow") + } + extraI = -1 - extraI + default: + return fmt.Errorf("wrong type for int64 field: %d", maj) + } + + t.QueuedBlocksTotal = int64(extraI) + } + // t.ReceivedBlocksTotal (int64) (int64) + case "ReceivedBlocksTotal": + { + maj, extra, err := cr.ReadHeader() + var extraI int64 + if err != nil { + return err + } + switch maj { + case cbg.MajUnsignedInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 positive overflow") + } + case cbg.MajNegativeInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 negative overflow") + } + extraI = -1 - extraI + default: + return fmt.Errorf("wrong type for int64 field: %d", maj) + } + + t.ReceivedBlocksTotal = int64(extraI) + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *EncodedVoucher) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.Type (datatransfer.TypeIdentifier) (string) + if len("Type") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Type\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Type"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Type")); err != nil { + return err + } + + if len(t.Type) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Type was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Type))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Type)); err != nil { + return err + } + + // t.Voucher (typegen.Deferred) (struct) + if len("Voucher") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Voucher\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Voucher"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Voucher")); err != nil { + return err + } + + if err := t.Voucher.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *EncodedVoucher) UnmarshalCBOR(r io.Reader) (err error) { + *t = EncodedVoucher{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("EncodedVoucher: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Type (datatransfer.TypeIdentifier) (string) + case "Type": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Type = datatransfer.TypeIdentifier(sval) + } + // t.Voucher (typegen.Deferred) (struct) + case "Voucher": + + { + + t.Voucher = new(cbg.Deferred) + + if err := t.Voucher.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("failed to read deferred field: %w", err) + } + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *EncodedVoucherResult) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.Type (datatransfer.TypeIdentifier) (string) + if len("Type") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Type\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Type"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Type")); err != nil { + return err + } + + if len(t.Type) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Type was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Type))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Type)); err != nil { + return err + } + + // t.VoucherResult (typegen.Deferred) (struct) + if len("VoucherResult") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"VoucherResult\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("VoucherResult"))); err != nil { + return err + } + if _, err := cw.WriteString(string("VoucherResult")); err != nil { + return err + } + + if err := t.VoucherResult.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *EncodedVoucherResult) UnmarshalCBOR(r io.Reader) (err error) { + *t = EncodedVoucherResult{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("EncodedVoucherResult: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Type (datatransfer.TypeIdentifier) (string) + case "Type": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Type = datatransfer.TypeIdentifier(sval) + } + // t.VoucherResult (typegen.Deferred) (struct) + case "VoucherResult": + + { + + t.VoucherResult = new(cbg.Deferred) + + if err := t.VoucherResult.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("failed to read deferred field: %w", err) + } + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} diff --git a/datatransfer/channels/internal/migrations/migrations.go b/datatransfer/channels/internal/migrations/migrations.go new file mode 100644 index 000000000..1f74b278e --- /dev/null +++ b/datatransfer/channels/internal/migrations/migrations.go @@ -0,0 +1,13 @@ +package migrations + +import ( + "github.com/libp2p/go-libp2p/core/peer" + + versioning "github.com/filecoin-project/go-ds-versioning/pkg" + "github.com/filecoin-project/go-ds-versioning/pkg/versioned" +) + +// GetChannelStateMigrations returns a migration list for the channel states +func GetChannelStateMigrations(selfPeer peer.ID) (versioning.VersionedMigrationList, error) { + return versioned.BuilderList{}.Build() +} diff --git a/datatransfer/encoding/encoding.go b/datatransfer/encoding/encoding.go new file mode 100644 index 000000000..dec7abcd7 --- /dev/null +++ b/datatransfer/encoding/encoding.go @@ -0,0 +1,171 @@ +package encoding + +import ( + "bytes" + "reflect" + + cbor "github.com/ipfs/go-ipld-cbor" + "github.com/ipld/go-ipld-prime" + "github.com/ipld/go-ipld-prime/codec/dagcbor" + "github.com/ipld/go-ipld-prime/datamodel" + "github.com/ipld/go-ipld-prime/node/basicnode" + "github.com/ipld/go-ipld-prime/schema" + cborgen "github.com/whyrusleeping/cbor-gen" + "golang.org/x/xerrors" +) + +// Encodable is an object that can be written to CBOR and decoded back +type Encodable interface{} + +// Encode encodes an encodable to CBOR, using the best available path for +// writing to CBOR +func Encode(value Encodable) ([]byte, error) { + if cbgEncodable, ok := value.(cborgen.CBORMarshaler); ok { + buf := new(bytes.Buffer) + err := cbgEncodable.MarshalCBOR(buf) + if err != nil { + return nil, err + } + return buf.Bytes(), nil + } + if ipldEncodable, ok := value.(datamodel.Node); ok { + if tn, ok := ipldEncodable.(schema.TypedNode); ok { + ipldEncodable = tn.Representation() + } + buf := &bytes.Buffer{} + err := dagcbor.Encode(ipldEncodable, buf) + if err != nil { + return nil, err + } + return buf.Bytes(), nil + } + return cbor.DumpObject(value) +} + +func EncodeToNode(encodable Encodable) (datamodel.Node, error) { + byts, err := Encode(encodable) + if err != nil { + return nil, err + } + na := basicnode.Prototype.Any.NewBuilder() + if err := dagcbor.Decode(na, bytes.NewReader(byts)); err != nil { + return nil, err + } + return na.Build(), nil +} + +// Decoder is CBOR decoder for a given encodable type +type Decoder interface { + DecodeFromCbor([]byte) (Encodable, error) + DecodeFromNode(datamodel.Node) (Encodable, error) +} + +// NewDecoder creates a new Decoder that will decode into new instances of the given +// object type. It will use the decoding that is optimal for that type +// It returns error if it's not possible to setup a decoder for this type +func NewDecoder(decodeType Encodable) (Decoder, error) { + // check if type is datamodel.Node, if so, just use style + if ipldDecodable, ok := decodeType.(datamodel.Node); ok { + return &ipldDecoder{ipldDecodable.Prototype()}, nil + } + // check if type is a pointer, as we need that to make new copies + // for cborgen types & regular IPLD types + decodeReflectType := reflect.TypeOf(decodeType) + if decodeReflectType.Kind() != reflect.Ptr { + return nil, xerrors.New("type must be a pointer") + } + // check if type is a cbor-gen type + if _, ok := decodeType.(cborgen.CBORUnmarshaler); ok { + return &cbgDecoder{decodeReflectType}, nil + } + // type does is neither ipld-prime nor cbor-gen, so we need to see if it + // can rountrip with oldschool ipld-format + encoded, err := cbor.DumpObject(decodeType) + if err != nil { + return nil, xerrors.New("Object type did not encode") + } + newDecodable := reflect.New(decodeReflectType.Elem()).Interface() + if err := cbor.DecodeInto(encoded, newDecodable); err != nil { + return nil, xerrors.New("Object type did not decode") + } + return &defaultDecoder{decodeReflectType}, nil +} + +type ipldDecoder struct { + style ipld.NodePrototype +} + +func (decoder *ipldDecoder) DecodeFromCbor(encoded []byte) (Encodable, error) { + builder := decoder.style.NewBuilder() + buf := bytes.NewReader(encoded) + err := dagcbor.Decode(builder, buf) + if err != nil { + return nil, err + } + return builder.Build(), nil +} + +func (decoder *ipldDecoder) DecodeFromNode(node datamodel.Node) (Encodable, error) { + builder := decoder.style.NewBuilder() + if err := builder.AssignNode(node); err != nil { + return nil, err + } + return builder.Build(), nil +} + +type cbgDecoder struct { + cbgType reflect.Type +} + +func (decoder *cbgDecoder) DecodeFromCbor(encoded []byte) (Encodable, error) { + decodedValue := reflect.New(decoder.cbgType.Elem()) + decoded, ok := decodedValue.Interface().(cborgen.CBORUnmarshaler) + if !ok || reflect.ValueOf(decoded).IsNil() { + return nil, xerrors.New("problem instantiating decoded value") + } + buf := bytes.NewReader(encoded) + err := decoded.UnmarshalCBOR(buf) + if err != nil { + return nil, err + } + return decoded, nil +} + +func (decoder *cbgDecoder) DecodeFromNode(node datamodel.Node) (Encodable, error) { + if tn, ok := node.(schema.TypedNode); ok { + node = tn.Representation() + } + buf := &bytes.Buffer{} + if err := dagcbor.Encode(node, buf); err != nil { + return nil, err + } + return decoder.DecodeFromCbor(buf.Bytes()) +} + +type defaultDecoder struct { + ptrType reflect.Type +} + +func (decoder *defaultDecoder) DecodeFromCbor(encoded []byte) (Encodable, error) { + decodedValue := reflect.New(decoder.ptrType.Elem()) + decoded, ok := decodedValue.Interface().(Encodable) + if !ok || reflect.ValueOf(decoded).IsNil() { + return nil, xerrors.New("problem instantiating decoded value") + } + err := cbor.DecodeInto(encoded, decoded) + if err != nil { + return nil, err + } + return decoded, nil +} + +func (decoder *defaultDecoder) DecodeFromNode(node datamodel.Node) (Encodable, error) { + if tn, ok := node.(schema.TypedNode); ok { + node = tn.Representation() + } + buf := &bytes.Buffer{} + if err := dagcbor.Encode(node, buf); err != nil { + return nil, err + } + return decoder.DecodeFromCbor(buf.Bytes()) +} diff --git a/datatransfer/errors.go b/datatransfer/errors.go new file mode 100644 index 000000000..0e9903f6d --- /dev/null +++ b/datatransfer/errors.go @@ -0,0 +1,32 @@ +package datatransfer + +type errorType string + +func (e errorType) Error() string { + return string(e) +} + +// ErrHandlerAlreadySet means an event handler was already set for this instance of +// hooks +const ErrHandlerAlreadySet = errorType("already set event handler") + +// ErrHandlerNotSet means you cannot issue commands to this interface because the +// handler has not been set +const ErrHandlerNotSet = errorType("event handler has not been set") + +// ErrChannelNotFound means the channel this command was issued for does not exist +const ErrChannelNotFound = errorType("channel not found") + +// ErrPause is a special error that the DataReceived / DataSent hooks can +// use to pause the channel +const ErrPause = errorType("pause channel") + +// ErrResume is a special error that the RequestReceived / ResponseReceived hooks can +// use to resume the channel +const ErrResume = errorType("resume channel") + +// ErrRejected indicates a request was not accepted +const ErrRejected = errorType("response rejected") + +// ErrUnsupported indicates an operation is not supported by the transport protocol +const ErrUnsupported = errorType("unsupported") diff --git a/datatransfer/events.go b/datatransfer/events.go new file mode 100644 index 000000000..664579c43 --- /dev/null +++ b/datatransfer/events.go @@ -0,0 +1,160 @@ +package datatransfer + +import "time" + +// EventCode is a name for an event that occurs on a data transfer channel +type EventCode int + +const ( + // Open is an event occurs when a channel is first opened + Open EventCode = iota + + // Accept is an event that emits when the data transfer is first accepted + Accept + + // Restart is an event that emits when the data transfer is restarted + Restart + + // DataReceived is emitted when data is received on the channel from a remote peer + DataReceived + + // DataSent is emitted when data is sent on the channel to the remote peer + DataSent + + // Cancel indicates one side has cancelled the transfer + Cancel + + // Error is an event that emits when an error occurs in a data transfer + Error + + // CleanupComplete emits when a request is cleaned up + CleanupComplete + + // NewVoucher means we have a new voucher on this channel + NewVoucher + + // NewVoucherResult means we have a new voucher result on this channel + NewVoucherResult + + // PauseInitiator emits when the data sender pauses transfer + PauseInitiator + + // ResumeInitiator emits when the data sender resumes transfer + ResumeInitiator + + // PauseResponder emits when the data receiver pauses transfer + PauseResponder + + // ResumeResponder emits when the data receiver resumes transfer + ResumeResponder + + // FinishTransfer emits when the initiator has completed sending/receiving data + FinishTransfer + + // ResponderCompletes emits when the initiator receives a message that the responder is finished + ResponderCompletes + + // ResponderBeginsFinalization emits when the initiator receives a message that the responder is finilizing + ResponderBeginsFinalization + + // BeginFinalizing emits when the responder completes its operations but awaits a response from the + // initiator + BeginFinalizing + + // Disconnected emits when we are not able to connect to the other party + Disconnected + + // Complete is emitted when a data transfer is complete + Complete + + // CompleteCleanupOnRestart is emitted when a data transfer channel is restarted to signal + // that channels that were cleaning up should finish cleanup + CompleteCleanupOnRestart + + // DataQueued is emitted when data is read and queued for sending to the remote peer + DataQueued + + // DataQueuedProgress is emitted when a block is queued for sending to the + // remote peer. It is not emitted when the block is resent. + // It is used to measure progress of how much of the total data has been + // queued. + DataQueuedProgress + + // DataSentProgress is emitted when a block is sent to the remote peer. + // It is not emitted when the block is resent. + // It is used to measure progress of how much of the total data has + // been sent. + DataSentProgress + + // DataReceivedProgress is emitted the first time a block is received from + // the remote peer. It is used to measure progress of how much of the total + // data has been received. + DataReceivedProgress + + // Deprecated in favour of RequestCancelled + RequestTimedOut + + // SendDataError indicates that the transport layer had an error trying + // to send data to the remote peer + SendDataError + + // ReceiveDataError indicates that the transport layer had an error + // receiving data from the remote peer + ReceiveDataError + + // TransferRequestQueued indicates that a new data transfer request has been queued in the transport layer + TransferRequestQueued + + // RequestCancelled indicates that a transport layer request was cancelled by the request opener + RequestCancelled + + // Opened is fired when a request for data is sent from this node to a peer + Opened +) + +// Events are human readable names for data transfer events +var Events = map[EventCode]string{ + Open: "Open", + Accept: "Accept", + Restart: "Restart", + DataReceived: "DataReceived", + DataSent: "DataSent", + Cancel: "Cancel", + Error: "Error", + CleanupComplete: "CleanupComplete", + NewVoucher: "NewVoucher", + NewVoucherResult: "NewVoucherResult", + PauseInitiator: "PauseInitiator", + ResumeInitiator: "ResumeInitiator", + PauseResponder: "PauseResponder", + ResumeResponder: "ResumeResponder", + FinishTransfer: "FinishTransfer", + ResponderCompletes: "ResponderCompletes", + ResponderBeginsFinalization: "ResponderBeginsFinalization", + BeginFinalizing: "BeginFinalizing", + Disconnected: "Disconnected", + Complete: "Complete", + CompleteCleanupOnRestart: "CompleteCleanupOnRestart", + DataQueued: "DataQueued", + DataQueuedProgress: "DataQueuedProgress", + DataSentProgress: "DataSentProgress", + DataReceivedProgress: "DataReceivedProgress", + RequestTimedOut: "RequestTimedOut", + SendDataError: "SendDataError", + ReceiveDataError: "ReceiveDataError", + TransferRequestQueued: "TransferRequestQueued", + RequestCancelled: "RequestCancelled", +} + +// Event is a struct containing information about a data transfer event +type Event struct { + Code EventCode // What type of event it is + Message string // Any clarifying information about the event + Timestamp time.Time // when the event happened +} + +// Subscriber is a callback that is called when events are emitted +type Subscriber func(event Event, channelState ChannelState) + +// Unsubscribe is a function that gets called to unsubscribe from data transfer events +type Unsubscribe func() diff --git a/datatransfer/impl/environment.go b/datatransfer/impl/environment.go new file mode 100644 index 000000000..51b95064a --- /dev/null +++ b/datatransfer/impl/environment.go @@ -0,0 +1,27 @@ +package impl + +import ( + "github.com/libp2p/go-libp2p/core/peer" + + "github.com/filecoin-project/boost/datatransfer" +) + +type channelEnvironment struct { + m *manager +} + +func (ce *channelEnvironment) Protect(id peer.ID, tag string) { + ce.m.dataTransferNetwork.Protect(id, tag) +} + +func (ce *channelEnvironment) Unprotect(id peer.ID, tag string) bool { + return ce.m.dataTransferNetwork.Unprotect(id, tag) +} + +func (ce *channelEnvironment) ID() peer.ID { + return ce.m.dataTransferNetwork.ID() +} + +func (ce *channelEnvironment) CleanupChannel(chid datatransfer.ChannelID) { + ce.m.transport.CleanupChannel(chid) +} diff --git a/datatransfer/impl/events.go b/datatransfer/impl/events.go new file mode 100644 index 000000000..f00b76a7b --- /dev/null +++ b/datatransfer/impl/events.go @@ -0,0 +1,590 @@ +package impl + +import ( + "context" + "errors" + + "github.com/ipfs/go-cid" + "github.com/ipld/go-ipld-prime" + cidlink "github.com/ipld/go-ipld-prime/linking/cid" + "github.com/libp2p/go-libp2p/core/peer" + "go.opentelemetry.io/otel" + "go.opentelemetry.io/otel/attribute" + "go.opentelemetry.io/otel/trace" + "golang.org/x/xerrors" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/encoding" + "github.com/filecoin-project/boost/datatransfer/registry" +) + +// OnChannelOpened is called when we send a request for data to the other +// peer on the given channel ID +func (m *manager) OnChannelOpened(chid datatransfer.ChannelID) error { + log.Infof("channel %s: opened", chid) + + // Check if the channel is being tracked + has, err := m.channels.HasChannel(chid) + if err != nil { + return err + } + if !has { + return datatransfer.ErrChannelNotFound + } + + // Fire an event + return m.channels.ChannelOpened(chid) +} + +// OnDataReceived is called when the transport layer reports that it has +// received some data from the sender. +// It fires an event on the channel, updating the sum of received data and +// calls revalidators so they can pause / resume the channel or send a +// message over the transport. +func (m *manager) OnDataReceived(chid datatransfer.ChannelID, link ipld.Link, size uint64, index int64, unique bool) error { + ctx, _ := m.spansIndex.SpanForChannel(context.TODO(), chid) + _, span := otel.Tracer("data-transfer").Start(ctx, "dataReceived", trace.WithAttributes( + attribute.String("channelID", chid.String()), + attribute.String("link", link.String()), + attribute.Int64("index", index), + attribute.Int64("size", int64(size)), + )) + defer span.End() + + isNew, err := m.channels.DataReceived(chid, link.(cidlink.Link).Cid, size, index, unique) + if err != nil { + return err + } + + // If this block has already been received on the channel, take no further + // action (this can happen when the data-transfer channel is restarted) + if !isNew { + return nil + } + + // If this node initiated the data transfer, there's nothing more to do + if chid.Initiator == m.peerID { + return nil + } + + // Check each revalidator to see if they want to pause / resume, or send + // a message over the transport + var result datatransfer.VoucherResult + var handled bool + _ = m.revalidators.Each(func(_ datatransfer.TypeIdentifier, _ encoding.Decoder, processor registry.Processor) error { + revalidator := processor.(datatransfer.Revalidator) + handled, result, err = revalidator.OnPushDataReceived(chid, size) + if handled { + return errors.New("stop processing") + } + return nil + }) + if err != nil || result != nil { + msg, err := m.processRevalidationResult(chid, result, err) + if msg != nil { + ctx, _ := m.spansIndex.SpanForChannel(context.TODO(), chid) + if err := m.dataTransferNetwork.SendMessage(ctx, chid.Initiator, msg); err != nil { + return err + } + } + return err + } + + return nil +} + +// OnDataQueued is called when the transport layer reports that it has queued +// up some data to be sent to the requester. +// It fires an event on the channel, updating the sum of queued data and calls +// revalidators so they can pause / resume or send a message over the transport. +func (m *manager) OnDataQueued(chid datatransfer.ChannelID, link ipld.Link, size uint64, index int64, unique bool) (datatransfer.Message, error) { + // The transport layer reports that some data has been queued up to be sent + // to the requester, so fire a DataQueued event on the channels state + // machine. + + ctx, _ := m.spansIndex.SpanForChannel(context.TODO(), chid) + _, span := otel.Tracer("data-transfer").Start(ctx, "dataQueued", trace.WithAttributes( + attribute.String("channelID", chid.String()), + attribute.String("link", link.String()), + attribute.Int64("size", int64(size)), + )) + defer span.End() + + isNew, err := m.channels.DataQueued(chid, link.(cidlink.Link).Cid, size, index, unique) + if err != nil { + return nil, err + } + + // If this block has already been queued on the channel, take no further + // action (this can happen when the data-transfer channel is restarted) + if !isNew { + return nil, nil + } + + // If this node initiated the data transfer, there's nothing more to do + if chid.Initiator == m.peerID { + return nil, nil + } + + // Check each revalidator to see if they want to pause / resume, or send + // a message over the transport. + // For example if the data-sender is waiting for the receiver to pay for + // data they may pause the data-transfer. + var result datatransfer.VoucherResult + var handled bool + _ = m.revalidators.Each(func(_ datatransfer.TypeIdentifier, _ encoding.Decoder, processor registry.Processor) error { + revalidator := processor.(datatransfer.Revalidator) + handled, result, err = revalidator.OnPullDataSent(chid, size) + if handled { + return errors.New("stop processing") + } + return nil + }) + if err != nil || result != nil { + return m.processRevalidationResult(chid, result, err) + } + + return nil, nil +} + +func (m *manager) OnDataSent(chid datatransfer.ChannelID, link ipld.Link, size uint64, index int64, unique bool) error { + + ctx, _ := m.spansIndex.SpanForChannel(context.TODO(), chid) + _, span := otel.Tracer("data-transfer").Start(ctx, "dataSent", trace.WithAttributes( + attribute.String("channelID", chid.String()), + attribute.String("link", link.String()), + attribute.Int64("size", int64(size)), + )) + defer span.End() + + _, err := m.channels.DataSent(chid, link.(cidlink.Link).Cid, size, index, unique) + return err +} + +func (m *manager) OnRequestReceived(chid datatransfer.ChannelID, request datatransfer.Request) (datatransfer.Response, error) { + if request.IsRestart() { + return m.receiveRestartRequest(chid, request) + } + + if request.IsNew() { + return m.receiveNewRequest(chid, request) + } + if request.IsCancel() { + log.Infof("channel %s: received cancel request, cleaning up channel", chid) + + m.transport.CleanupChannel(chid) + return nil, m.channels.Cancel(chid) + } + if request.IsVoucher() { + return m.processUpdateVoucher(chid, request) + } + if request.IsPaused() { + return nil, m.pauseOther(chid) + } + err := m.resumeOther(chid) + if err != nil { + return nil, err + } + chst, err := m.channels.GetByID(context.TODO(), chid) + if err != nil { + return nil, err + } + if chst.Status() == datatransfer.ResponderPaused || + chst.Status() == datatransfer.ResponderFinalizing { + return nil, datatransfer.ErrPause + } + return nil, nil +} + +func (m *manager) OnTransferQueued(chid datatransfer.ChannelID) { + m.channels.TransferRequestQueued(chid) //nolint:errcheck +} + +func (m *manager) OnResponseReceived(chid datatransfer.ChannelID, response datatransfer.Response) error { + if response.IsComplete() { + log.Infow("received complete response", "chid", chid, "isAccepted", response.Accepted()) + } + + if response.IsCancel() { + log.Infof("channel %s: received cancel response, cancelling channel", chid) + return m.channels.Cancel(chid) + } + if response.IsVoucherResult() { + if !response.EmptyVoucherResult() { + vresult, err := m.decodeVoucherResult(response) + if err != nil { + return err + } + err = m.channels.NewVoucherResult(chid, vresult) + if err != nil { + return err + } + } + if !response.Accepted() { + log.Infof("channel %s: received rejected response, erroring out channel", chid) + return m.channels.Error(chid, datatransfer.ErrRejected) + } + if response.IsNew() { + log.Infof("channel %s: received new response, accepting channel", chid) + err := m.channels.Accept(chid) + if err != nil { + return err + } + } + + if response.IsRestart() { + log.Infof("channel %s: received restart response, restarting channel", chid) + err := m.channels.Restart(chid) + if err != nil { + return err + } + } + } + if response.IsComplete() && response.Accepted() { + if !response.IsPaused() { + log.Infow("received complete response,responder not paused, completing channel", "chid", chid) + return m.channels.ResponderCompletes(chid) + } + + log.Infow("received complete response, responder is paused, not completing channel", "chid", chid) + err := m.channels.ResponderBeginsFinalization(chid) + if err != nil { + return nil + } + } + if response.IsPaused() { + return m.pauseOther(chid) + } + return m.resumeOther(chid) +} + +func (m *manager) OnRequestCancelled(chid datatransfer.ChannelID, err error) error { + log.Warnf("channel %+v was cancelled: %s", chid, err) + return m.channels.RequestCancelled(chid, err) +} + +func (m *manager) OnRequestDisconnected(chid datatransfer.ChannelID, err error) error { + log.Warnf("channel %+v has stalled or disconnected: %s", chid, err) + return m.channels.Disconnected(chid, err) +} + +func (m *manager) OnSendDataError(chid datatransfer.ChannelID, err error) error { + log.Debugf("channel %+v had transport send error: %s", chid, err) + return m.channels.SendDataError(chid, err) +} + +func (m *manager) OnReceiveDataError(chid datatransfer.ChannelID, err error) error { + log.Debugf("channel %+v had transport receive error: %s", chid, err) + return m.channels.ReceiveDataError(chid, err) +} + +// OnChannelCompleted is called +// - by the requester when all data for a transfer has been received +// - by the responder when all data for a transfer has been sent +func (m *manager) OnChannelCompleted(chid datatransfer.ChannelID, completeErr error) error { + // If the channel completed successfully + if completeErr == nil { + // If the channel was initiated by the other peer + if chid.Initiator != m.peerID { + log.Infow("received OnChannelCompleted, will send completion message to initiator", "chid", chid) + msg, err := m.completeMessage(chid) + if err != nil { + return err + } + if msg != nil { + // Send the other peer a message that the transfer has completed + log.Infow("sending completion message to initiator", "chid", chid) + ctx, _ := m.spansIndex.SpanForChannel(context.Background(), chid) + if err := m.dataTransferNetwork.SendMessage(ctx, chid.Initiator, msg); err != nil { + err := xerrors.Errorf("channel %s: failed to send completion message to initiator: %w", chid, err) + log.Warnw("failed to send completion message to initiator", "chid", chid, "err", err) + return m.OnRequestDisconnected(chid, err) + } + log.Infow("successfully sent completion message to initiator", "chid", chid) + } + if msg.Accepted() { + if msg.IsPaused() { + return m.channels.BeginFinalizing(chid) + } + return m.channels.Complete(chid) + } + return m.channels.Error(chid, err) + } + + // The channel was initiated by this node, so move to the finished state + log.Infof("channel %s: transfer initiated by local node is complete", chid) + return m.channels.FinishTransfer(chid) + } + + // There was an error so fire an Error event + chst, err := m.channels.GetByID(context.TODO(), chid) + if err != nil { + return err + } + // send an error, but only if we haven't already errored for some reason + if chst.Status() != datatransfer.Failing && chst.Status() != datatransfer.Failed { + err := xerrors.Errorf("data transfer channel %s failed to transfer data: %w", chid, completeErr) + log.Warnf(err.Error()) + return m.channels.Error(chid, err) + } + return nil +} + +func (m *manager) OnContextAugment(chid datatransfer.ChannelID) func(context.Context) context.Context { + return func(ctx context.Context) context.Context { + ctx, _ = m.spansIndex.SpanForChannel(ctx, chid) + return ctx + } +} + +func (m *manager) receiveRestartRequest(chid datatransfer.ChannelID, incoming datatransfer.Request) (datatransfer.Response, error) { + log.Infof("channel %s: received restart request", chid) + + result, err := m.restartRequest(chid, incoming) + msg, msgErr := m.response(true, false, err, incoming.TransferID(), result) + if msgErr != nil { + return nil, msgErr + } + return msg, err +} + +func (m *manager) receiveNewRequest(chid datatransfer.ChannelID, incoming datatransfer.Request) (datatransfer.Response, error) { + log.Infof("channel %s: received new channel request from %s", chid, chid.Initiator) + + result, err := m.acceptRequest(chid, incoming) + msg, msgErr := m.response(false, true, err, incoming.TransferID(), result) + if msgErr != nil { + return nil, msgErr + } + return msg, err +} + +func (m *manager) restartRequest(chid datatransfer.ChannelID, + incoming datatransfer.Request) (datatransfer.VoucherResult, error) { + + initiator := chid.Initiator + if m.peerID == initiator { + return nil, xerrors.New("initiator cannot be manager peer for a restart request") + } + + if err := m.validateRestartRequest(context.Background(), initiator, chid, incoming); err != nil { + return nil, xerrors.Errorf("restart request for channel %s failed validation: %w", chid, err) + } + + stor, err := incoming.Selector() + if err != nil { + return nil, err + } + + voucher, result, err := m.validateVoucher(true, chid, initiator, incoming, incoming.IsPull(), incoming.BaseCid(), stor) + if err != nil && err != datatransfer.ErrPause { + return result, xerrors.Errorf("failed to validate voucher: %w", err) + } + voucherErr := err + + if result != nil { + err := m.channels.NewVoucherResult(chid, result) + if err != nil { + return result, err + } + } + if err := m.channels.Restart(chid); err != nil { + return result, xerrors.Errorf("failed to restart channel %s: %w", chid, err) + } + processor, has := m.transportConfigurers.Processor(voucher.Type()) + if has { + transportConfigurer := processor.(datatransfer.TransportConfigurer) + transportConfigurer(chid, voucher, m.transport) + } + m.dataTransferNetwork.Protect(initiator, chid.String()) + if voucherErr == datatransfer.ErrPause { + err := m.channels.PauseResponder(chid) + if err != nil { + return result, err + } + } + return result, voucherErr +} + +func (m *manager) acceptRequest(chid datatransfer.ChannelID, incoming datatransfer.Request) (datatransfer.VoucherResult, error) { + + stor, err := incoming.Selector() + if err != nil { + return nil, err + } + + voucher, result, err := m.validateVoucher(false, chid, chid.Initiator, incoming, incoming.IsPull(), incoming.BaseCid(), stor) + if err != nil && err != datatransfer.ErrPause { + return result, err + } + voucherErr := err + + var dataSender, dataReceiver peer.ID + if incoming.IsPull() { + dataSender = m.peerID + dataReceiver = chid.Initiator + } else { + dataSender = chid.Initiator + dataReceiver = m.peerID + } + + log.Infow("data-transfer request validated, will create & start tracking channel", "channelID", chid, "payloadCid", incoming.BaseCid()) + _, err = m.channels.CreateNew(m.peerID, incoming.TransferID(), incoming.BaseCid(), stor, voucher, chid.Initiator, dataSender, dataReceiver) + if err != nil { + log.Errorw("failed to create and start tracking channel", "channelID", chid, "err", err) + return result, err + } + log.Debugw("successfully created and started tracking channel", "channelID", chid) + if result != nil { + err := m.channels.NewVoucherResult(chid, result) + if err != nil { + return result, err + } + } + if err := m.channels.Accept(chid); err != nil { + return result, err + } + processor, has := m.transportConfigurers.Processor(voucher.Type()) + if has { + transportConfigurer := processor.(datatransfer.TransportConfigurer) + transportConfigurer(chid, voucher, m.transport) + } + m.dataTransferNetwork.Protect(chid.Initiator, chid.String()) + if voucherErr == datatransfer.ErrPause { + err := m.channels.PauseResponder(chid) + if err != nil { + return result, err + } + } + return result, voucherErr +} + +// validateVoucher converts a voucher in an incoming message to its appropriate +// voucher struct, then runs the validator and returns the results. +// returns error if: +// - reading voucher fails +// - deserialization of selector fails +// - validation fails +func (m *manager) validateVoucher( + isRestart bool, + chid datatransfer.ChannelID, + sender peer.ID, + incoming datatransfer.Request, + isPull bool, + baseCid cid.Cid, + stor ipld.Node, +) (datatransfer.Voucher, datatransfer.VoucherResult, error) { + vouch, err := m.decodeVoucher(incoming, m.validatedTypes) + if err != nil { + return nil, nil, err + } + var validatorFunc func(bool, datatransfer.ChannelID, peer.ID, datatransfer.Voucher, cid.Cid, ipld.Node) (datatransfer.VoucherResult, error) + processor, _ := m.validatedTypes.Processor(vouch.Type()) + validator := processor.(datatransfer.RequestValidator) + if isPull { + validatorFunc = validator.ValidatePull + } else { + validatorFunc = validator.ValidatePush + } + + result, err := validatorFunc(isRestart, chid, sender, vouch, baseCid, stor) + return vouch, result, err +} + +// revalidateVoucher converts a voucher in an incoming message to its appropriate +// voucher struct, then runs the revalidator and returns the results. +// returns error if: +// - reading voucher fails +// - deserialization of selector fails +// - validation fails +func (m *manager) revalidateVoucher(chid datatransfer.ChannelID, + incoming datatransfer.Request) (datatransfer.Voucher, datatransfer.VoucherResult, error) { + vouch, err := m.decodeVoucher(incoming, m.revalidators) + if err != nil { + return nil, nil, err + } + processor, _ := m.revalidators.Processor(vouch.Type()) + validator := processor.(datatransfer.Revalidator) + + result, err := validator.Revalidate(chid, vouch) + return vouch, result, err +} + +func (m *manager) processUpdateVoucher(chid datatransfer.ChannelID, request datatransfer.Request) (datatransfer.Response, error) { + vouch, result, voucherErr := m.revalidateVoucher(chid, request) + if vouch != nil { + err := m.channels.NewVoucher(chid, vouch) + if err != nil { + return nil, err + } + } + return m.processRevalidationResult(chid, result, voucherErr) +} + +func (m *manager) revalidationResponse(chid datatransfer.ChannelID, result datatransfer.VoucherResult, resultErr error) (datatransfer.Response, error) { + chst, err := m.channels.GetByID(context.TODO(), chid) + if err != nil { + return nil, err + } + if chst.Status() == datatransfer.Finalizing { + return m.completeResponse(resultErr, chid.ID, result) + } + return m.response(false, false, resultErr, chid.ID, result) +} + +func (m *manager) processRevalidationResult(chid datatransfer.ChannelID, result datatransfer.VoucherResult, resultErr error) (datatransfer.Response, error) { + vresMessage, err := m.revalidationResponse(chid, result, resultErr) + + if err != nil { + return nil, err + } + if result != nil { + err := m.channels.NewVoucherResult(chid, result) + if err != nil { + return nil, err + } + } + + if resultErr == nil { + return vresMessage, nil + } + + if resultErr == datatransfer.ErrPause { + err := m.pause(chid) + if err != nil { + return nil, err + } + return vresMessage, datatransfer.ErrPause + } + + if resultErr == datatransfer.ErrResume { + err = m.resume(chid) + if err != nil { + return nil, err + } + return vresMessage, datatransfer.ErrResume + } + return vresMessage, resultErr +} + +func (m *manager) completeMessage(chid datatransfer.ChannelID) (datatransfer.Response, error) { + var result datatransfer.VoucherResult + var resultErr error + var handled bool + _ = m.revalidators.Each(func(_ datatransfer.TypeIdentifier, _ encoding.Decoder, processor registry.Processor) error { + revalidator := processor.(datatransfer.Revalidator) + handled, result, resultErr = revalidator.OnComplete(chid) + if handled { + return errors.New("stop processing") + } + return nil + }) + if result != nil { + err := m.channels.NewVoucherResult(chid, result) + if err != nil { + return nil, err + } + } + + return m.completeResponse(resultErr, chid.ID, result) +} diff --git a/datatransfer/impl/impl.go b/datatransfer/impl/impl.go new file mode 100644 index 000000000..c58ef8fa1 --- /dev/null +++ b/datatransfer/impl/impl.go @@ -0,0 +1,561 @@ +package impl + +import ( + "context" + "errors" + "fmt" + "time" + + "github.com/hannahhoward/go-pubsub" + "github.com/ipfs/go-cid" + "github.com/ipfs/go-datastore" + logging "github.com/ipfs/go-log/v2" + "github.com/ipld/go-ipld-prime" + cidlink "github.com/ipld/go-ipld-prime/linking/cid" + "github.com/libp2p/go-libp2p/core/peer" + "go.opentelemetry.io/otel" + "go.opentelemetry.io/otel/attribute" + "go.opentelemetry.io/otel/codes" + "go.opentelemetry.io/otel/trace" + "golang.org/x/xerrors" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/channelmonitor" + "github.com/filecoin-project/boost/datatransfer/channels" + "github.com/filecoin-project/boost/datatransfer/encoding" + "github.com/filecoin-project/boost/datatransfer/message" + "github.com/filecoin-project/boost/datatransfer/network" + "github.com/filecoin-project/boost/datatransfer/registry" + "github.com/filecoin-project/boost/datatransfer/tracing" +) + +var log = logging.Logger("dt-impl") +var cancelSendTimeout = 30 * time.Second + +type manager struct { + dataTransferNetwork network.DataTransferNetwork + validatedTypes *registry.Registry + resultTypes *registry.Registry + revalidators *registry.Registry + transportConfigurers *registry.Registry + pubSub *pubsub.PubSub + readySub *pubsub.PubSub + channels *channels.Channels + peerID peer.ID + transport datatransfer.Transport + channelMonitor *channelmonitor.Monitor + channelMonitorCfg *channelmonitor.Config + transferIDGen *timeCounter + spansIndex *tracing.SpansIndex +} + +type internalEvent struct { + evt datatransfer.Event + state datatransfer.ChannelState +} + +func dispatcher(evt pubsub.Event, subscriberFn pubsub.SubscriberFn) error { + ie, ok := evt.(internalEvent) + if !ok { + return errors.New("wrong type of event") + } + cb, ok := subscriberFn.(datatransfer.Subscriber) + if !ok { + return errors.New("wrong type of event") + } + cb(ie.evt, ie.state) + return nil +} + +func readyDispatcher(evt pubsub.Event, fn pubsub.SubscriberFn) error { + migrateErr, ok := evt.(error) + if !ok && evt != nil { + return errors.New("wrong type of event") + } + cb, ok := fn.(datatransfer.ReadyFunc) + if !ok { + return errors.New("wrong type of event") + } + cb(migrateErr) + return nil +} + +// DataTransferOption configures the data transfer manager +type DataTransferOption func(*manager) + +// ChannelRestartConfig sets the configuration options for automatically +// restarting push and pull channels +func ChannelRestartConfig(cfg channelmonitor.Config) DataTransferOption { + return func(m *manager) { + m.channelMonitorCfg = &cfg + } +} + +// NewDataTransfer initializes a new instance of a data transfer manager +func NewDataTransfer(ds datastore.Batching, dataTransferNetwork network.DataTransferNetwork, transport datatransfer.Transport, options ...DataTransferOption) (datatransfer.Manager, error) { + m := &manager{ + dataTransferNetwork: dataTransferNetwork, + validatedTypes: registry.NewRegistry(), + resultTypes: registry.NewRegistry(), + revalidators: registry.NewRegistry(), + transportConfigurers: registry.NewRegistry(), + pubSub: pubsub.New(dispatcher), + readySub: pubsub.New(readyDispatcher), + peerID: dataTransferNetwork.ID(), + transport: transport, + transferIDGen: newTimeCounter(), + spansIndex: tracing.NewSpansIndex(), + } + + channels, err := channels.New(ds, m.notifier, m.voucherDecoder, m.resultTypes.Decoder, &channelEnvironment{m}, dataTransferNetwork.ID()) + if err != nil { + return nil, err + } + m.channels = channels + + // Apply config options + for _, option := range options { + option(m) + } + + // Create push / pull channel monitor after applying config options as the config + // options may apply to the monitor + m.channelMonitor = channelmonitor.NewMonitor(m, m.channelMonitorCfg) + + return m, nil +} + +func (m *manager) voucherDecoder(voucherType datatransfer.TypeIdentifier) (encoding.Decoder, bool) { + decoder, has := m.validatedTypes.Decoder(voucherType) + if !has { + return m.revalidators.Decoder(voucherType) + } + return decoder, true +} + +func (m *manager) notifier(evt datatransfer.Event, chst datatransfer.ChannelState) { + err := m.pubSub.Publish(internalEvent{evt, chst}) + if err != nil { + log.Warnf("err publishing DT event: %s", err.Error()) + } +} + +// Start initializes data transfer processing +func (m *manager) Start(ctx context.Context) error { + log.Info("start data-transfer module") + + go func() { + err := m.channels.Start(ctx) + if err != nil { + log.Errorf("Migrating data transfer state machines: %s", err.Error()) + } + err = m.readySub.Publish(err) + if err != nil { + log.Warnf("Publish data transfer ready event: %s", err.Error()) + } + }() + + dtReceiver := &receiver{m} + m.dataTransferNetwork.SetDelegate(dtReceiver) + return m.transport.SetEventHandler(m) +} + +// OnReady registers a listener for when the data transfer manager has finished starting up +func (m *manager) OnReady(ready datatransfer.ReadyFunc) { + m.readySub.Subscribe(ready) +} + +// Stop terminates all data transfers and ends processing +func (m *manager) Stop(ctx context.Context) error { + log.Info("stop data-transfer module") + m.channelMonitor.Shutdown() + m.spansIndex.EndAll() + return m.transport.Shutdown(ctx) +} + +// RegisterVoucherType registers a validator for the given voucher type +// returns error if: +// * voucher type does not implement voucher +// * there is a voucher type registered with an identical identifier +// * voucherType's Kind is not reflect.Ptr +func (m *manager) RegisterVoucherType(voucherType datatransfer.Voucher, validator datatransfer.RequestValidator) error { + err := m.validatedTypes.Register(voucherType, validator) + if err != nil { + return xerrors.Errorf("error registering voucher type: %w", err) + } + return nil +} + +// OpenPushDataChannel opens a data transfer that will send data to the recipient peer and +// transfer parts of the piece that match the selector +func (m *manager) OpenPushDataChannel(ctx context.Context, requestTo peer.ID, voucher datatransfer.Voucher, baseCid cid.Cid, selector ipld.Node) (datatransfer.ChannelID, error) { + log.Infof("open push channel to %s with base cid %s", requestTo, baseCid) + + req, err := m.newRequest(ctx, selector, false, voucher, baseCid, requestTo) + if err != nil { + return datatransfer.ChannelID{}, err + } + + chid, err := m.channels.CreateNew(m.peerID, req.TransferID(), baseCid, selector, voucher, + m.peerID, m.peerID, requestTo) // initiator = us, sender = us, receiver = them + if err != nil { + return chid, err + } + ctx, span := m.spansIndex.SpanForChannel(ctx, chid) + processor, has := m.transportConfigurers.Processor(voucher.Type()) + if has { + transportConfigurer := processor.(datatransfer.TransportConfigurer) + transportConfigurer(chid, voucher, m.transport) + } + m.dataTransferNetwork.Protect(requestTo, chid.String()) + monitoredChan := m.channelMonitor.AddPushChannel(chid) + if err := m.dataTransferNetwork.SendMessage(ctx, requestTo, req); err != nil { + err = fmt.Errorf("Unable to send request: %w", err) + span.RecordError(err) + span.SetStatus(codes.Error, err.Error()) + _ = m.channels.Error(chid, err) + + // If push channel monitoring is enabled, shutdown the monitor as it + // wasn't possible to start the data transfer + if monitoredChan != nil { + monitoredChan.Shutdown() + } + + return chid, err + } + + return chid, nil +} + +// OpenPullDataChannel opens a data transfer that will request data from the sending peer and +// transfer parts of the piece that match the selector +func (m *manager) OpenPullDataChannel(ctx context.Context, requestTo peer.ID, voucher datatransfer.Voucher, baseCid cid.Cid, selector ipld.Node) (datatransfer.ChannelID, error) { + log.Infof("open pull channel to %s with base cid %s", requestTo, baseCid) + + req, err := m.newRequest(ctx, selector, true, voucher, baseCid, requestTo) + if err != nil { + return datatransfer.ChannelID{}, err + } + // initiator = us, sender = them, receiver = us + chid, err := m.channels.CreateNew(m.peerID, req.TransferID(), baseCid, selector, voucher, + m.peerID, requestTo, m.peerID) + if err != nil { + return chid, err + } + ctx, span := m.spansIndex.SpanForChannel(ctx, chid) + processor, has := m.transportConfigurers.Processor(voucher.Type()) + if has { + transportConfigurer := processor.(datatransfer.TransportConfigurer) + transportConfigurer(chid, voucher, m.transport) + } + m.dataTransferNetwork.Protect(requestTo, chid.String()) + monitoredChan := m.channelMonitor.AddPullChannel(chid) + if err := m.transport.OpenChannel(ctx, requestTo, chid, cidlink.Link{Cid: baseCid}, selector, nil, req); err != nil { + err = fmt.Errorf("Unable to send request: %w", err) + span.RecordError(err) + span.SetStatus(codes.Error, err.Error()) + _ = m.channels.Error(chid, err) + + // If pull channel monitoring is enabled, shutdown the monitor as it + // wasn't possible to start the data transfer + if monitoredChan != nil { + monitoredChan.Shutdown() + } + return chid, err + } + return chid, nil +} + +// SendVoucher sends an intermediate voucher as needed when the receiver sends a request for revalidation +func (m *manager) SendVoucher(ctx context.Context, channelID datatransfer.ChannelID, voucher datatransfer.Voucher) error { + chst, err := m.channels.GetByID(ctx, channelID) + if err != nil { + return err + } + ctx, _ = m.spansIndex.SpanForChannel(ctx, channelID) + ctx, span := otel.Tracer("data-transfer").Start(ctx, "sendVoucher", trace.WithAttributes( + attribute.String("channelID", channelID.String()), + attribute.String("voucherType", string(voucher.Type())), + )) + defer span.End() + if channelID.Initiator != m.peerID { + err := errors.New("cannot send voucher for request we did not initiate") + span.RecordError(err) + span.SetStatus(codes.Error, err.Error()) + return err + } + updateRequest, err := message.VoucherRequest(channelID.ID, voucher.Type(), voucher) + if err != nil { + span.RecordError(err) + span.SetStatus(codes.Error, err.Error()) + return err + } + if err := m.dataTransferNetwork.SendMessage(ctx, chst.OtherPeer(), updateRequest); err != nil { + err = fmt.Errorf("Unable to send request: %w", err) + _ = m.OnRequestDisconnected(channelID, err) + span.RecordError(err) + span.SetStatus(codes.Error, err.Error()) + return err + } + return m.channels.NewVoucher(channelID, voucher) +} + +// close an open channel (effectively a cancel) +func (m *manager) CloseDataTransferChannel(ctx context.Context, chid datatransfer.ChannelID) error { + log.Infof("close channel %s", chid) + + chst, err := m.channels.GetByID(ctx, chid) + if err != nil { + return err + } + ctx, _ = m.spansIndex.SpanForChannel(ctx, chid) + ctx, span := otel.Tracer("data-transfer").Start(ctx, "closeChannel", trace.WithAttributes( + attribute.String("channelID", chid.String()), + )) + defer span.End() + // Close the channel on the local transport + err = m.transport.CloseChannel(ctx, chid) + if err != nil { + span.RecordError(err) + span.SetStatus(codes.Error, err.Error()) + log.Warnf("unable to close channel %s: %s", chid, err) + } + + // Send a cancel message to the remote peer async + go func() { + sctx, cancel := context.WithTimeout(context.Background(), cancelSendTimeout) + defer cancel() + log.Infof("%s: sending cancel channel to %s for channel %s", m.peerID, chst.OtherPeer(), chid) + err = m.dataTransferNetwork.SendMessage(sctx, chst.OtherPeer(), m.cancelMessage(chid)) + if err != nil { + err = fmt.Errorf("unable to send cancel message for channel %s to peer %s: %w", + chid, m.peerID, err) + _ = m.OnRequestDisconnected(chid, err) + log.Warn(err) + } + }() + + // Fire a cancel event + fsmerr := m.channels.Cancel(chid) + if fsmerr != nil { + return xerrors.Errorf("unable to send cancel to channel FSM: %w", fsmerr) + } + + return nil +} + +// ConnectTo opens a connection to a peer on the data-transfer protocol, +// retrying if necessary +func (m *manager) ConnectTo(ctx context.Context, p peer.ID) error { + return m.dataTransferNetwork.ConnectWithRetry(ctx, p) +} + +// close an open channel and fire an error event +func (m *manager) CloseDataTransferChannelWithError(ctx context.Context, chid datatransfer.ChannelID, cherr error) error { + log.Infof("close channel %s with error %s", chid, cherr) + + chst, err := m.channels.GetByID(ctx, chid) + if err != nil { + return err + } + ctx, _ = m.spansIndex.SpanForChannel(ctx, chid) + ctx, span := otel.Tracer("data-transfer").Start(ctx, "closeChannel", trace.WithAttributes( + attribute.String("channelID", chid.String()), + )) + defer span.End() + + // Cancel the channel on the local transport + err = m.transport.CloseChannel(ctx, chid) + if err != nil { + log.Warnf("unable to close channel %s: %s", chid, err) + } + + // Try to send a cancel message to the remote peer. It's quite likely + // we aren't able to send the message to the peer because the channel + // is already in an error state, which is probably because of connection + // issues, so if we cant send the message just log a warning. + log.Infof("%s: sending cancel channel to %s for channel %s", m.peerID, chst.OtherPeer(), chid) + err = m.dataTransferNetwork.SendMessage(ctx, chst.OtherPeer(), m.cancelMessage(chid)) + if err != nil { + // Just log a warning here because it's important that we fire the + // error event with the original error so that it doesn't get masked + // by subsequent errors. + log.Warnf("unable to send cancel message for channel %s to peer %s: %w", + chid, m.peerID, err) + } + + // Fire an error event + err = m.channels.Error(chid, cherr) + if err != nil { + return xerrors.Errorf("unable to send error %s to channel FSM: %w", cherr, err) + } + + return nil +} + +// pause a running data transfer channel +func (m *manager) PauseDataTransferChannel(ctx context.Context, chid datatransfer.ChannelID) error { + log.Infof("pause channel %s", chid) + + pausable, ok := m.transport.(datatransfer.PauseableTransport) + if !ok { + return datatransfer.ErrUnsupported + } + + ctx, _ = m.spansIndex.SpanForChannel(ctx, chid) + + err := pausable.PauseChannel(ctx, chid) + if err != nil { + log.Warnf("Error attempting to pause at transport level: %s", err.Error()) + } + + if err := m.dataTransferNetwork.SendMessage(ctx, chid.OtherParty(m.peerID), m.pauseMessage(chid)); err != nil { + err = fmt.Errorf("Unable to send pause message: %w", err) + _ = m.OnRequestDisconnected(chid, err) + return err + } + + return m.pause(chid) +} + +// resume a running data transfer channel +func (m *manager) ResumeDataTransferChannel(ctx context.Context, chid datatransfer.ChannelID) error { + log.Infof("resume channel %s", chid) + + pausable, ok := m.transport.(datatransfer.PauseableTransport) + if !ok { + return datatransfer.ErrUnsupported + } + + ctx, _ = m.spansIndex.SpanForChannel(ctx, chid) + + err := pausable.ResumeChannel(ctx, m.resumeMessage(chid), chid) + if err != nil { + log.Warnf("Error attempting to resume at transport level: %s", err.Error()) + } + + return m.resume(chid) +} + +// get channel state +func (m *manager) ChannelState(ctx context.Context, chid datatransfer.ChannelID) (datatransfer.ChannelState, error) { + return m.channels.GetByID(ctx, chid) +} + +// get status of a transfer +func (m *manager) TransferChannelStatus(ctx context.Context, chid datatransfer.ChannelID) datatransfer.Status { + chst, err := m.channels.GetByID(ctx, chid) + if err != nil { + return datatransfer.ChannelNotFoundError + } + return chst.Status() +} + +// get notified when certain types of events happen +func (m *manager) SubscribeToEvents(subscriber datatransfer.Subscriber) datatransfer.Unsubscribe { + return datatransfer.Unsubscribe(m.pubSub.Subscribe(subscriber)) +} + +// get all in progress transfers +func (m *manager) InProgressChannels(ctx context.Context) (map[datatransfer.ChannelID]datatransfer.ChannelState, error) { + return m.channels.InProgress() +} + +// RegisterRevalidator registers a revalidator for the given voucher type +// Note: this is the voucher type used to revalidate. It can share a name +// with the initial validator type and CAN be the same type, or a different type. +// The revalidator can simply be the sampe as the original request validator, +// or a different validator that satisfies the revalidator interface. +func (m *manager) RegisterRevalidator(voucherType datatransfer.Voucher, revalidator datatransfer.Revalidator) error { + err := m.revalidators.Register(voucherType, revalidator) + if err != nil { + return xerrors.Errorf("error registering revalidator type: %w", err) + } + return nil +} + +// RegisterVoucherResultType allows deserialization of a voucher result, +// so that a listener can read the metadata +func (m *manager) RegisterVoucherResultType(resultType datatransfer.VoucherResult) error { + err := m.resultTypes.Register(resultType, nil) + if err != nil { + return xerrors.Errorf("error registering voucher type: %w", err) + } + return nil +} + +// RegisterTransportConfigurer registers the given transport configurer to be run on requests with the given voucher +// type +func (m *manager) RegisterTransportConfigurer(voucherType datatransfer.Voucher, configurer datatransfer.TransportConfigurer) error { + err := m.transportConfigurers.Register(voucherType, configurer) + if err != nil { + return xerrors.Errorf("error registering transport configurer: %w", err) + } + return nil +} + +// RestartDataTransferChannel restarts data transfer on the channel with the given channelId +func (m *manager) RestartDataTransferChannel(ctx context.Context, chid datatransfer.ChannelID) error { + log.Infof("restart channel %s", chid) + + channel, err := m.channels.GetByID(ctx, chid) + if err != nil { + return xerrors.Errorf("failed to fetch channel: %w", err) + } + + // if channel has already been completed, there is nothing to do. + // TODO We could be in a state where the channel has completed but the corresponding event hasnt fired in the client/provider. + if channels.IsChannelTerminated(channel.Status()) { + return nil + } + + // if channel is is cleanup state, finish it + if channels.IsChannelCleaningUp(channel.Status()) { + return m.channels.CompleteCleanupOnRestart(channel.ChannelID()) + } + + ctx, _ = m.spansIndex.SpanForChannel(ctx, chid) + ctx, span := otel.Tracer("data-transfer").Start(ctx, "restartChannel", trace.WithAttributes( + attribute.String("channelID", chid.String()), + )) + defer span.End() + // initiate restart + chType := m.channelDataTransferType(channel) + switch chType { + case ManagerPeerReceivePush: + return m.restartManagerPeerReceivePush(ctx, channel) + case ManagerPeerReceivePull: + return m.restartManagerPeerReceivePull(ctx, channel) + case ManagerPeerCreatePull: + return m.openPullRestartChannel(ctx, channel) + case ManagerPeerCreatePush: + return m.openPushRestartChannel(ctx, channel) + } + + return nil +} + +func (m *manager) channelDataTransferType(channel datatransfer.ChannelState) ChannelDataTransferType { + initiator := channel.ChannelID().Initiator + if channel.IsPull() { + // we created a pull channel + if initiator == m.peerID { + return ManagerPeerCreatePull + } + + // we received a pull channel + return ManagerPeerReceivePull + } + + // we created a push channel + if initiator == m.peerID { + return ManagerPeerCreatePush + } + + // we received a push channel + return ManagerPeerReceivePush +} + +func (m *manager) PeerID() peer.ID { + return m.peerID +} diff --git a/datatransfer/impl/receiver.go b/datatransfer/impl/receiver.go new file mode 100644 index 000000000..02f0402b9 --- /dev/null +++ b/datatransfer/impl/receiver.go @@ -0,0 +1,192 @@ +package impl + +import ( + "context" + + cidlink "github.com/ipld/go-ipld-prime/linking/cid" + "github.com/libp2p/go-libp2p/core/peer" + "go.opentelemetry.io/otel" + "go.opentelemetry.io/otel/attribute" + "go.opentelemetry.io/otel/trace" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/channels" +) + +type receiver struct { + manager *manager +} + +// ReceiveRequest takes an incoming data transfer request, validates the voucher and +// processes the message. +func (r *receiver) ReceiveRequest( + ctx context.Context, + initiator peer.ID, + incoming datatransfer.Request) { + err := r.receiveRequest(ctx, initiator, incoming) + if err != nil { + log.Warnf("error processing request from %s: %s", initiator, err) + } +} + +func (r *receiver) receiveRequest(ctx context.Context, initiator peer.ID, incoming datatransfer.Request) error { + chid := datatransfer.ChannelID{Initiator: initiator, Responder: r.manager.peerID, ID: incoming.TransferID()} + ctx, _ = r.manager.spansIndex.SpanForChannel(ctx, chid) + ctx, span := otel.Tracer("data-transfer").Start(ctx, "receiveRequest", trace.WithAttributes( + attribute.String("channelID", chid.String()), + attribute.String("baseCid", incoming.BaseCid().String()), + attribute.Bool("isNew", incoming.IsNew()), + attribute.Bool("isRestart", incoming.IsRestart()), + attribute.Bool("isUpdate", incoming.IsUpdate()), + attribute.Bool("isCancel", incoming.IsCancel()), + attribute.Bool("isPaused", incoming.IsPaused()), + )) + defer span.End() + response, receiveErr := r.manager.OnRequestReceived(chid, incoming) + + if receiveErr == datatransfer.ErrResume { + chst, err := r.manager.channels.GetByID(ctx, chid) + if err != nil { + return err + } + if resumeTransportStatesResponder.Contains(chst.Status()) { + return r.manager.transport.(datatransfer.PauseableTransport).ResumeChannel(ctx, response, chid) + } + receiveErr = nil + } + + if response != nil { + if (response.IsNew() || response.IsRestart()) && response.Accepted() && !incoming.IsPull() { + var channel datatransfer.ChannelState + if response.IsRestart() { + var err error + channel, err = r.manager.channels.GetByID(ctx, chid) + if err != nil { + return err + } + } + + stor, _ := incoming.Selector() + if err := r.manager.transport.OpenChannel(ctx, initiator, chid, cidlink.Link{Cid: incoming.BaseCid()}, stor, channel, response); err != nil { + return err + } + } else { + if err := r.manager.dataTransferNetwork.SendMessage(ctx, initiator, response); err != nil { + return err + } + } + } + + if receiveErr == datatransfer.ErrPause { + return r.manager.transport.(datatransfer.PauseableTransport).PauseChannel(ctx, chid) + } + + if receiveErr != nil { + _ = r.manager.transport.CloseChannel(ctx, chid) + return receiveErr + } + + return nil +} + +// ReceiveResponse handles responses to our Push or Pull data transfer request. +// It schedules a transfer only if our Pull Request is accepted. +func (r *receiver) ReceiveResponse( + ctx context.Context, + sender peer.ID, + incoming datatransfer.Response) { + err := r.receiveResponse(ctx, sender, incoming) + if err != nil { + log.Error(err) + } +} +func (r *receiver) receiveResponse( + ctx context.Context, + sender peer.ID, + incoming datatransfer.Response) error { + chid := datatransfer.ChannelID{Initiator: r.manager.peerID, Responder: sender, ID: incoming.TransferID()} + ctx, _ = r.manager.spansIndex.SpanForChannel(ctx, chid) + ctx, span := otel.Tracer("data-transfer").Start(ctx, "receiveResponse", trace.WithAttributes( + attribute.String("channelID", chid.String()), + attribute.Bool("accepted", incoming.Accepted()), + attribute.Bool("isComplete", incoming.IsComplete()), + attribute.Bool("isNew", incoming.IsNew()), + attribute.Bool("isRestart", incoming.IsRestart()), + attribute.Bool("isUpdate", incoming.IsUpdate()), + attribute.Bool("isCancel", incoming.IsCancel()), + attribute.Bool("isPaused", incoming.IsPaused()), + )) + defer span.End() + err := r.manager.OnResponseReceived(chid, incoming) + if err == datatransfer.ErrPause { + return r.manager.transport.(datatransfer.PauseableTransport).PauseChannel(ctx, chid) + } + if err != nil { + log.Warnf("closing channel %s after getting error processing response from %s: %s", + chid, sender, err) + + _ = r.manager.transport.CloseChannel(ctx, chid) + return err + } + return nil +} + +func (r *receiver) ReceiveError(err error) { + log.Errorf("received error message on data transfer: %s", err.Error()) +} + +func (r *receiver) ReceiveRestartExistingChannelRequest(ctx context.Context, + sender peer.ID, + incoming datatransfer.Request) { + + ch, err := incoming.RestartChannelId() + if err != nil { + log.Errorf("cannot restart channel: failed to fetch channel Id: %w", err) + return + } + + ctx, _ = r.manager.spansIndex.SpanForChannel(ctx, ch) + ctx, span := otel.Tracer("data-transfer").Start(ctx, "receiveRequest", trace.WithAttributes( + attribute.String("channelID", ch.String()), + )) + defer span.End() + log.Infof("channel %s: received restart existing channel request from %s", ch, sender) + + // validate channel exists -> in non-terminal state and that the sender matches + channel, err := r.manager.channels.GetByID(ctx, ch) + if err != nil || channel == nil { + // nothing to do here, we wont handle the request + return + } + + // initiator should be me + if channel.ChannelID().Initiator != r.manager.peerID { + log.Errorf("cannot restart channel %s: channel initiator is not the manager peer", ch) + return + } + + // other peer should be the counter party on the channel + if channel.OtherPeer() != sender { + log.Errorf("cannot restart channel %s: channel counterparty is not the sender peer", ch) + return + } + + // channel should NOT be terminated + if channels.IsChannelTerminated(channel.Status()) { + log.Errorf("cannot restart channel %s: channel already terminated", ch) + return + } + + switch r.manager.channelDataTransferType(channel) { + case ManagerPeerCreatePush: + if err := r.manager.openPushRestartChannel(ctx, channel); err != nil { + log.Errorf("failed to open push restart channel %s: %s", ch, err) + } + case ManagerPeerCreatePull: + if err := r.manager.openPullRestartChannel(ctx, channel); err != nil { + log.Errorf("failed to open pull restart channel %s: %s", ch, err) + } + default: + log.Error("peer is not the creator of the channel") + } +} diff --git a/datatransfer/impl/restart.go b/datatransfer/impl/restart.go new file mode 100644 index 000000000..29aaf0404 --- /dev/null +++ b/datatransfer/impl/restart.go @@ -0,0 +1,198 @@ +package impl + +import ( + "bytes" + "context" + + cidlink "github.com/ipld/go-ipld-prime/linking/cid" + "github.com/libp2p/go-libp2p/core/peer" + "golang.org/x/xerrors" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/channels" + "github.com/filecoin-project/boost/datatransfer/encoding" + "github.com/filecoin-project/boost/datatransfer/message" +) + +// ChannelDataTransferType identifies the type of a data transfer channel for the purposes of a restart +type ChannelDataTransferType int + +const ( + // ManagerPeerCreatePull is the type of a channel wherein the manager peer created a Pull Data Transfer + ManagerPeerCreatePull ChannelDataTransferType = iota + + // ManagerPeerCreatePush is the type of a channel wherein the manager peer created a Push Data Transfer + ManagerPeerCreatePush + + // ManagerPeerReceivePull is the type of a channel wherein the manager peer received a Pull Data Transfer Request + ManagerPeerReceivePull + + // ManagerPeerReceivePush is the type of a channel wherein the manager peer received a Push Data Transfer Request + ManagerPeerReceivePush +) + +func (m *manager) restartManagerPeerReceivePush(ctx context.Context, channel datatransfer.ChannelState) error { + if err := m.validateRestartVoucher(channel, false); err != nil { + return xerrors.Errorf("failed to restart channel, validation error: %w", err) + } + + // send a libp2p message to the other peer asking to send a "restart push request" + req := message.RestartExistingChannelRequest(channel.ChannelID()) + + if err := m.dataTransferNetwork.SendMessage(ctx, channel.OtherPeer(), req); err != nil { + return xerrors.Errorf("unable to send restart request: %w", err) + } + + return nil +} + +func (m *manager) restartManagerPeerReceivePull(ctx context.Context, channel datatransfer.ChannelState) error { + if err := m.validateRestartVoucher(channel, true); err != nil { + return xerrors.Errorf("failed to restart channel, validation error: %w", err) + } + + req := message.RestartExistingChannelRequest(channel.ChannelID()) + + // send a libp2p message to the other peer asking to send a "restart pull request" + if err := m.dataTransferNetwork.SendMessage(ctx, channel.OtherPeer(), req); err != nil { + return xerrors.Errorf("unable to send restart request: %w", err) + } + + return nil +} + +func (m *manager) validateRestartVoucher(channel datatransfer.ChannelState, isPull bool) error { + // re-validate the original voucher received for safety + chid := channel.ChannelID() + + // recreate the request that would have led to this pull channel being created for validation + req, err := message.NewRequest(chid.ID, false, isPull, channel.Voucher().Type(), channel.Voucher(), + channel.BaseCID(), channel.Selector()) + if err != nil { + return err + } + + // revalidate the voucher by reconstructing the request that would have led to the creation of this channel + if _, _, err := m.validateVoucher(true, chid, channel.OtherPeer(), req, isPull, channel.BaseCID(), channel.Selector()); err != nil { + return err + } + + return nil +} + +func (m *manager) openPushRestartChannel(ctx context.Context, channel datatransfer.ChannelState) error { + selector := channel.Selector() + voucher := channel.Voucher() + baseCid := channel.BaseCID() + requestTo := channel.OtherPeer() + chid := channel.ChannelID() + + req, err := message.NewRequest(chid.ID, true, false, voucher.Type(), voucher, baseCid, selector) + if err != nil { + return err + } + + processor, has := m.transportConfigurers.Processor(voucher.Type()) + if has { + transportConfigurer := processor.(datatransfer.TransportConfigurer) + transportConfigurer(chid, voucher, m.transport) + } + m.dataTransferNetwork.Protect(requestTo, chid.String()) + + // Monitor the state of the connection for the channel + monitoredChan := m.channelMonitor.AddPushChannel(chid) + log.Infof("sending push restart channel to %s for channel %s", requestTo, chid) + if err := m.dataTransferNetwork.SendMessage(ctx, requestTo, req); err != nil { + // If push channel monitoring is enabled, shutdown the monitor as it + // wasn't possible to start the data transfer + if monitoredChan != nil { + monitoredChan.Shutdown() + } + + return xerrors.Errorf("Unable to send restart request: %w", err) + } + + return nil +} + +func (m *manager) openPullRestartChannel(ctx context.Context, channel datatransfer.ChannelState) error { + selector := channel.Selector() + voucher := channel.Voucher() + baseCid := channel.BaseCID() + requestTo := channel.OtherPeer() + chid := channel.ChannelID() + + req, err := message.NewRequest(chid.ID, true, true, voucher.Type(), voucher, baseCid, selector) + if err != nil { + return err + } + + processor, has := m.transportConfigurers.Processor(voucher.Type()) + if has { + transportConfigurer := processor.(datatransfer.TransportConfigurer) + transportConfigurer(chid, voucher, m.transport) + } + m.dataTransferNetwork.Protect(requestTo, chid.String()) + + // Monitor the state of the connection for the channel + monitoredChan := m.channelMonitor.AddPullChannel(chid) + log.Infof("sending open channel to %s to restart channel %s", requestTo, chid) + if err := m.transport.OpenChannel(ctx, requestTo, chid, cidlink.Link{Cid: baseCid}, selector, channel, req); err != nil { + // If pull channel monitoring is enabled, shutdown the monitor as it + // wasn't possible to start the data transfer + if monitoredChan != nil { + monitoredChan.Shutdown() + } + + return xerrors.Errorf("Unable to send open channel restart request: %w", err) + } + + return nil +} + +func (m *manager) validateRestartRequest(ctx context.Context, otherPeer peer.ID, chid datatransfer.ChannelID, req datatransfer.Request) error { + // channel should exist + channel, err := m.channels.GetByID(ctx, chid) + if err != nil { + return err + } + + // channel is not terminated + if channels.IsChannelTerminated(channel.Status()) { + return xerrors.New("channel is already terminated") + } + + // channel initator should be the sender peer + if channel.ChannelID().Initiator != otherPeer { + return xerrors.New("other peer is not the initiator of the channel") + } + + // channel and request baseCid should match + if req.BaseCid() != channel.BaseCID() { + return xerrors.New("base cid does not match") + } + + // vouchers should match + reqVoucher, err := m.decodeVoucher(req, m.validatedTypes) + if err != nil { + return xerrors.Errorf("failed to decode request voucher: %w", err) + } + if reqVoucher.Type() != channel.Voucher().Type() { + return xerrors.New("channel and request voucher types do not match") + } + + reqBz, err := encoding.Encode(reqVoucher) + if err != nil { + return xerrors.New("failed to encode request voucher") + } + channelBz, err := encoding.Encode(channel.Voucher()) + if err != nil { + return xerrors.New("failed to encode channel voucher") + } + + if !bytes.Equal(reqBz, channelBz) { + return xerrors.New("channel and request vouchers do not match") + } + + return nil +} diff --git a/datatransfer/impl/timecounter.go b/datatransfer/impl/timecounter.go new file mode 100644 index 000000000..c13904f07 --- /dev/null +++ b/datatransfer/impl/timecounter.go @@ -0,0 +1,21 @@ +package impl + +import ( + "sync/atomic" + "time" +) + +// timeCounter is used to generate a monotonically increasing sequence. +// It starts at the current time, then increments on each call to next. +type timeCounter struct { + counter uint64 +} + +func newTimeCounter() *timeCounter { + return &timeCounter{counter: uint64(time.Now().UnixNano())} +} + +func (tc *timeCounter) next() uint64 { + counter := atomic.AddUint64(&tc.counter, 1) + return counter +} diff --git a/datatransfer/impl/utils.go b/datatransfer/impl/utils.go new file mode 100644 index 000000000..37842a43c --- /dev/null +++ b/datatransfer/impl/utils.go @@ -0,0 +1,140 @@ +package impl + +import ( + "context" + + "github.com/ipfs/go-cid" + "github.com/ipld/go-ipld-prime" + "github.com/libp2p/go-libp2p/core/peer" + "golang.org/x/xerrors" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/message" + "github.com/filecoin-project/boost/datatransfer/registry" +) + +type statusList []datatransfer.Status + +func (sl statusList) Contains(s datatransfer.Status) bool { + for _, ts := range sl { + if ts == s { + return true + } + } + return false +} + +var resumeTransportStatesResponder = statusList{ + datatransfer.Requested, + datatransfer.Ongoing, + datatransfer.InitiatorPaused, +} + +// newRequest encapsulates message creation +func (m *manager) newRequest(ctx context.Context, selector ipld.Node, isPull bool, voucher datatransfer.Voucher, baseCid cid.Cid, to peer.ID) (datatransfer.Request, error) { + // Generate a new transfer ID for the request + tid := datatransfer.TransferID(m.transferIDGen.next()) + return message.NewRequest(tid, false, isPull, voucher.Type(), voucher, baseCid, selector) +} + +func (m *manager) response(isRestart bool, isNew bool, err error, tid datatransfer.TransferID, voucherResult datatransfer.VoucherResult) (datatransfer.Response, error) { + isAccepted := err == nil || err == datatransfer.ErrPause || err == datatransfer.ErrResume + isPaused := err == datatransfer.ErrPause + resultType := datatransfer.EmptyTypeIdentifier + if voucherResult != nil { + resultType = voucherResult.Type() + } + if isRestart { + return message.RestartResponse(tid, isAccepted, isPaused, resultType, voucherResult) + } + + if isNew { + return message.NewResponse(tid, isAccepted, isPaused, resultType, voucherResult) + } + return message.VoucherResultResponse(tid, isAccepted, isPaused, resultType, voucherResult) +} + +func (m *manager) completeResponse(err error, tid datatransfer.TransferID, voucherResult datatransfer.VoucherResult) (datatransfer.Response, error) { + isAccepted := err == nil || err == datatransfer.ErrPause || err == datatransfer.ErrResume + isPaused := err == datatransfer.ErrPause + resultType := datatransfer.EmptyTypeIdentifier + if voucherResult != nil { + resultType = voucherResult.Type() + } + return message.CompleteResponse(tid, isAccepted, isPaused, resultType, voucherResult) +} + +func (m *manager) resume(chid datatransfer.ChannelID) error { + if chid.Initiator == m.peerID { + return m.channels.ResumeInitiator(chid) + } + return m.channels.ResumeResponder(chid) +} + +func (m *manager) pause(chid datatransfer.ChannelID) error { + if chid.Initiator == m.peerID { + return m.channels.PauseInitiator(chid) + } + return m.channels.PauseResponder(chid) +} + +func (m *manager) resumeOther(chid datatransfer.ChannelID) error { + if chid.Responder == m.peerID { + return m.channels.ResumeInitiator(chid) + } + return m.channels.ResumeResponder(chid) +} + +func (m *manager) pauseOther(chid datatransfer.ChannelID) error { + if chid.Responder == m.peerID { + return m.channels.PauseInitiator(chid) + } + return m.channels.PauseResponder(chid) +} + +func (m *manager) resumeMessage(chid datatransfer.ChannelID) datatransfer.Message { + if chid.Initiator == m.peerID { + return message.UpdateRequest(chid.ID, false) + } + return message.UpdateResponse(chid.ID, false) +} + +func (m *manager) pauseMessage(chid datatransfer.ChannelID) datatransfer.Message { + if chid.Initiator == m.peerID { + return message.UpdateRequest(chid.ID, true) + } + return message.UpdateResponse(chid.ID, true) +} + +func (m *manager) cancelMessage(chid datatransfer.ChannelID) datatransfer.Message { + if chid.Initiator == m.peerID { + return message.CancelRequest(chid.ID) + } + return message.CancelResponse(chid.ID) +} + +func (m *manager) decodeVoucherResult(response datatransfer.Response) (datatransfer.VoucherResult, error) { + vtypStr := datatransfer.TypeIdentifier(response.VoucherResultType()) + decoder, has := m.resultTypes.Decoder(vtypStr) + if !has { + return nil, xerrors.Errorf("unknown voucher result type: %s", vtypStr) + } + encodable, err := response.VoucherResult(decoder) + if err != nil { + return nil, err + } + return encodable.(datatransfer.Registerable), nil +} + +func (m *manager) decodeVoucher(request datatransfer.Request, registry *registry.Registry) (datatransfer.Voucher, error) { + vtypStr := datatransfer.TypeIdentifier(request.VoucherType()) + decoder, has := registry.Decoder(vtypStr) + if !has { + return nil, xerrors.Errorf("unknown voucher type: %s", vtypStr) + } + encodable, err := request.Voucher(decoder) + if err != nil { + return nil, err + } + return encodable.(datatransfer.Registerable), nil +} diff --git a/datatransfer/manager.go b/datatransfer/manager.go new file mode 100644 index 000000000..37dc7afe4 --- /dev/null +++ b/datatransfer/manager.go @@ -0,0 +1,260 @@ +package datatransfer + +import ( + "context" + + "github.com/ipfs/go-cid" + "github.com/ipld/go-ipld-prime" + "github.com/libp2p/go-libp2p/core/peer" +) + +// RequestValidator is an interface implemented by the client of the +// data transfer module to validate requests +type RequestValidator interface { + // ValidatePush validates a push request received from the peer that will send data + ValidatePush( + isRestart bool, + chid ChannelID, + sender peer.ID, + voucher Voucher, + baseCid cid.Cid, + selector ipld.Node) (VoucherResult, error) + // ValidatePull validates a pull request received from the peer that will receive data + ValidatePull( + isRestart bool, + chid ChannelID, + receiver peer.ID, + voucher Voucher, + baseCid cid.Cid, + selector ipld.Node) (VoucherResult, error) +} + +// Revalidator is a request validator revalidates in progress requests +// by requesting request additional vouchers, and resuming when it receives them +type Revalidator interface { + // Revalidate revalidates a request with a new voucher + Revalidate(channelID ChannelID, voucher Voucher) (VoucherResult, error) + // OnPullDataSent is called on the responder side when more bytes are sent + // for a given pull request. The first value indicates whether the request was + // recognized by this revalidator and should be considered 'handled'. If true, + // the remaining two values are interpreted. If 'false' the request is passed on + // to the next revalidators. + // It should return a VoucherResult + ErrPause to + // request revalidation or nil to continue uninterrupted, + // other errors will terminate the request. + OnPullDataSent(chid ChannelID, additionalBytesSent uint64) (bool, VoucherResult, error) + // OnPushDataReceived is called on the responder side when more bytes are received + // for a given push request. The first value indicates whether the request was + // recognized by this revalidator and should be considered 'handled'. If true, + // the remaining two values are interpreted. If 'false' the request is passed on + // to the next revalidators. It should return a VoucherResult + ErrPause to + // request revalidation or nil to continue uninterrupted, + // other errors will terminate the request + OnPushDataReceived(chid ChannelID, additionalBytesReceived uint64) (bool, VoucherResult, error) + // OnComplete is called to make a final request for revalidation -- often for the + // purpose of settlement. The first value indicates whether the request was + // recognized by this revalidator and should be considered 'handled'. If true, + // the remaining two values are interpreted. If 'false' the request is passed on + // to the next revalidators. + // if VoucherResult is non nil, the request will enter a settlement phase awaiting + // a final update + OnComplete(chid ChannelID) (bool, VoucherResult, error) +} + +// TransportConfigurer provides a mechanism to provide transport specific configuration for a given voucher type +type TransportConfigurer func(chid ChannelID, voucher Voucher, transport Transport) + +// ReadyFunc is function that gets called once when the data transfer module is ready +type ReadyFunc func(error) + +// Manager is the core interface presented by all implementations of +// of the data transfer sub system +type Manager interface { + + // Start initializes data transfer processing + Start(ctx context.Context) error + + // OnReady registers a listener for when the data transfer comes on line + OnReady(ReadyFunc) + + // Stop terminates all data transfers and ends processing + Stop(ctx context.Context) error + + // RegisterVoucherType registers a validator for the given voucher type + // will error if voucher type does not implement voucher + // or if there is a voucher type registered with an identical identifier + RegisterVoucherType(voucherType Voucher, validator RequestValidator) error + + // RegisterRevalidator registers a revalidator for the given voucher type + // Note: this is the voucher type used to revalidate. It can share a name + // with the initial validator type and CAN be the same type, or a different type. + // The revalidator can simply be the sampe as the original request validator, + // or a different validator that satisfies the revalidator interface. + RegisterRevalidator(voucherType Voucher, revalidator Revalidator) error + + // RegisterVoucherResultType allows deserialization of a voucher result, + // so that a listener can read the metadata + RegisterVoucherResultType(resultType VoucherResult) error + + // RegisterTransportConfigurer registers the given transport configurer to be run on requests with the given voucher + // type + RegisterTransportConfigurer(voucherType Voucher, configurer TransportConfigurer) error + + // open a data transfer that will send data to the recipient peer and + // transfer parts of the piece that match the selector + OpenPushDataChannel(ctx context.Context, to peer.ID, voucher Voucher, baseCid cid.Cid, selector ipld.Node) (ChannelID, error) + + // open a data transfer that will request data from the sending peer and + // transfer parts of the piece that match the selector + OpenPullDataChannel(ctx context.Context, to peer.ID, voucher Voucher, baseCid cid.Cid, selector ipld.Node) (ChannelID, error) + + // send an intermediate voucher as needed when the receiver sends a request for revalidation + SendVoucher(ctx context.Context, chid ChannelID, voucher Voucher) error + + // close an open channel (effectively a cancel) + CloseDataTransferChannel(ctx context.Context, chid ChannelID) error + + // pause a data transfer channel (only allowed if transport supports it) + PauseDataTransferChannel(ctx context.Context, chid ChannelID) error + + // resume a data transfer channel (only allowed if transport supports it) + ResumeDataTransferChannel(ctx context.Context, chid ChannelID) error + + // get status of a transfer + TransferChannelStatus(ctx context.Context, x ChannelID) Status + + // get channel state + ChannelState(ctx context.Context, chid ChannelID) (ChannelState, error) + + // get notified when certain types of events happen + SubscribeToEvents(subscriber Subscriber) Unsubscribe + + // get all in progress transfers + InProgressChannels(ctx context.Context) (map[ChannelID]ChannelState, error) + + // RestartDataTransferChannel restarts an existing data transfer channel + RestartDataTransferChannel(ctx context.Context, chid ChannelID) error +} + +/* +Transport defines the interface for a transport layer for data +transfer. Where the data transfer manager will coordinate setting up push and +pull requests, validation, etc, the transport layer is responsible for moving +data back and forth, and may be medium specific. For example, some transports +may have the ability to pause and resume requests, while others may not. +Some may support individual data events, while others may only support message +events. Some transport layers may opt to use the actual data transfer network +protocols directly while others may be able to encode messages in their own +data protocol. + +Transport is the minimum interface that must be satisfied to serve as a datatransfer +transport layer. Transports must be able to open (open is always called by the receiving peer) +and close channels, and set at an event handler +*/ +type Transport interface { + // OpenChannel initiates an outgoing request for the other peer to send data + // to us on this channel + // Note: from a data transfer symantic standpoint, it doesn't matter if the + // request is push or pull -- OpenChannel is called by the party that is + // intending to receive data + OpenChannel( + ctx context.Context, + dataSender peer.ID, + channelID ChannelID, + root ipld.Link, + stor ipld.Node, + channel ChannelState, + msg Message, + ) error + + // CloseChannel closes the given channel + CloseChannel(ctx context.Context, chid ChannelID) error + // SetEventHandler sets the handler for events on channels + SetEventHandler(events EventsHandler) error + // CleanupChannel is called on the otherside of a cancel - removes any associated + // data for the channel + CleanupChannel(chid ChannelID) + Shutdown(ctx context.Context) error +} + +// EventsHandler are semantic data transfer events that happen as a result of graphsync hooks +type EventsHandler interface { + // OnChannelOpened is called when we send a request for data to the other + // peer on the given channel ID + // return values are: + // - error = ignore incoming data for this channel + OnChannelOpened(chid ChannelID) error + // OnResponseReceived is called when we receive a response to a request + // - nil = continue receiving data + // - error = cancel this request + OnResponseReceived(chid ChannelID, msg Response) error + // OnDataReceive is called when we receive data for the given channel ID + // return values are: + // - nil = proceed with sending data + // - error = cancel this request + // - err == ErrPause - pause this request + OnDataReceived(chid ChannelID, link ipld.Link, size uint64, index int64, unique bool) error + + // OnDataQueued is called when data is queued for sending for the given channel ID + // return values are: + // message = data transfer message along with data + // err = error + // - nil = proceed with sending data + // - error = cancel this request + // - err == ErrPause - pause this request + OnDataQueued(chid ChannelID, link ipld.Link, size uint64, index int64, unique bool) (Message, error) + + // OnDataSent is called when we send data for the given channel ID + OnDataSent(chid ChannelID, link ipld.Link, size uint64, index int64, unique bool) error + + // OnTransferQueued is called when a new data transfer request is queued in the transport layer. + OnTransferQueued(chid ChannelID) + + // OnRequestReceived is called when we receive a new request to send data + // for the given channel ID + // return values are: + // message = data transfer message along with reply + // err = error + // - nil = proceed with sending data + // - error = cancel this request + // - err == ErrPause - pause this request (only for new requests) + // - err == ErrResume - resume this request (only for update requests) + OnRequestReceived(chid ChannelID, msg Request) (Response, error) + // OnChannelCompleted is called when we finish transferring data for the given channel ID + // Error returns are logged but otherwise have no effect + OnChannelCompleted(chid ChannelID, err error) error + + // OnRequestCancelled is called when a request we opened (with the given channel Id) to + // receive data is cancelled by us. + // Error returns are logged but otherwise have no effect + OnRequestCancelled(chid ChannelID, err error) error + + // OnRequestDisconnected is called when a network error occurs trying to send a request + OnRequestDisconnected(chid ChannelID, err error) error + + // OnSendDataError is called when a network error occurs sending data + // at the transport layer + OnSendDataError(chid ChannelID, err error) error + + // OnReceiveDataError is called when a network error occurs receiving data + // at the transport layer + OnReceiveDataError(chid ChannelID, err error) error + + // OnContextAugment allows the transport to attach data transfer tracing information + // to its local context, in order to create a hierarchical trace + OnContextAugment(chid ChannelID) func(context.Context) context.Context +} + +// PauseableTransport is a transport that can also pause and resume channels +type PauseableTransport interface { + Transport + // PauseChannel paused the given channel ID + PauseChannel(ctx context.Context, + chid ChannelID, + ) error + // ResumeChannel resumes the given channel + ResumeChannel(ctx context.Context, + msg Message, + chid ChannelID, + ) error +} diff --git a/datatransfer/message.go b/datatransfer/message.go new file mode 100644 index 000000000..371ba86a0 --- /dev/null +++ b/datatransfer/message.go @@ -0,0 +1,56 @@ +package datatransfer + +import ( + "io" + + "github.com/filecoin-project/boost/datatransfer/encoding" + "github.com/ipfs/go-cid" + "github.com/ipld/go-ipld-prime" + "github.com/ipld/go-ipld-prime/datamodel" + "github.com/libp2p/go-libp2p/core/protocol" +) + +var ( + // ProtocolDataTransfer1_2 is the protocol identifier for the latest + // version of data-transfer (supports do-not-send-first-blocks extension) + ProtocolDataTransfer1_2 protocol.ID = "/fil/datatransfer/1.2.0" +) + +// Message is a message for the data transfer protocol +// (either request or response) that can serialize to a protobuf +type Message interface { + IsRequest() bool + IsRestart() bool + IsNew() bool + IsUpdate() bool + IsPaused() bool + IsCancel() bool + TransferID() TransferID + ToNet(w io.Writer) error + ToIPLD() (datamodel.Node, error) + MessageForProtocol(targetProtocol protocol.ID) (newMsg Message, err error) +} + +// Request is a response message for the data transfer protocol +type Request interface { + Message + IsPull() bool + IsVoucher() bool + VoucherType() TypeIdentifier + Voucher(decoder encoding.Decoder) (encoding.Encodable, error) + BaseCid() cid.Cid + Selector() (ipld.Node, error) + IsRestartExistingChannelRequest() bool + RestartChannelId() (ChannelID, error) +} + +// Response is a response message for the data transfer protocol +type Response interface { + Message + IsVoucherResult() bool + IsComplete() bool + Accepted() bool + VoucherResultType() TypeIdentifier + VoucherResult(decoder encoding.Decoder) (encoding.Encodable, error) + EmptyVoucherResult() bool +} diff --git a/datatransfer/message/message.go b/datatransfer/message/message.go new file mode 100644 index 000000000..f98cb3a3f --- /dev/null +++ b/datatransfer/message/message.go @@ -0,0 +1,19 @@ +package message + +import ( + "github.com/filecoin-project/boost/datatransfer/message/message1_1prime" +) + +var NewRequest = message1_1.NewRequest +var RestartExistingChannelRequest = message1_1.RestartExistingChannelRequest +var UpdateRequest = message1_1.UpdateRequest +var VoucherRequest = message1_1.VoucherRequest +var RestartResponse = message1_1.RestartResponse +var NewResponse = message1_1.NewResponse +var VoucherResultResponse = message1_1.VoucherResultResponse +var CancelResponse = message1_1.CancelResponse +var UpdateResponse = message1_1.UpdateResponse +var FromNet = message1_1.FromNet +var FromIPLD = message1_1.FromIPLD +var CompleteResponse = message1_1.CompleteResponse +var CancelRequest = message1_1.CancelRequest diff --git a/datatransfer/message/message1_1/message.go b/datatransfer/message/message1_1/message.go new file mode 100644 index 000000000..c79de18a1 --- /dev/null +++ b/datatransfer/message/message1_1/message.go @@ -0,0 +1,195 @@ +package message1_1 + +import ( + "bytes" + "io" + + datatransfer2 "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/encoding" + "github.com/filecoin-project/boost/datatransfer/message/types" + "github.com/ipfs/go-cid" + "github.com/ipld/go-ipld-prime" + "github.com/ipld/go-ipld-prime/codec/dagcbor" + "github.com/ipld/go-ipld-prime/datamodel" + cborgen "github.com/whyrusleeping/cbor-gen" + "golang.org/x/xerrors" +) + +// NewRequest generates a new request for the data transfer protocol +func NewRequest(id datatransfer2.TransferID, isRestart bool, isPull bool, vtype datatransfer2.TypeIdentifier, voucher encoding.Encodable, baseCid cid.Cid, selector ipld.Node) (datatransfer2.Request, error) { + vbytes, err := encoding.Encode(voucher) + if err != nil { + return nil, xerrors.Errorf("Creating request: %w", err) + } + if baseCid == cid.Undef { + return nil, xerrors.Errorf("base CID must be defined") + } + selBytes, err := encoding.Encode(selector) + if err != nil { + return nil, xerrors.Errorf("Error encoding selector") + } + + var typ uint64 + if isRestart { + typ = uint64(types.RestartMessage) + } else { + typ = uint64(types.NewMessage) + } + + return &TransferRequest1_1{ + Type: typ, + Pull: isPull, + Vouch: &cborgen.Deferred{Raw: vbytes}, + Stor: &cborgen.Deferred{Raw: selBytes}, + BCid: &baseCid, + VTyp: vtype, + XferID: uint64(id), + }, nil +} + +// RestartExistingChannelRequest creates a request to ask the other side to restart an existing channel +func RestartExistingChannelRequest(channelId datatransfer2.ChannelID) datatransfer2.Request { + + return &TransferRequest1_1{Type: uint64(types.RestartExistingChannelRequestMessage), + RestartChannel: channelId} +} + +// CancelRequest request generates a request to cancel an in progress request +func CancelRequest(id datatransfer2.TransferID) datatransfer2.Request { + return &TransferRequest1_1{ + Type: uint64(types.CancelMessage), + XferID: uint64(id), + } +} + +// UpdateRequest generates a new request update +func UpdateRequest(id datatransfer2.TransferID, isPaused bool) datatransfer2.Request { + return &TransferRequest1_1{ + Type: uint64(types.UpdateMessage), + Paus: isPaused, + XferID: uint64(id), + } +} + +// VoucherRequest generates a new request for the data transfer protocol +func VoucherRequest(id datatransfer2.TransferID, vtype datatransfer2.TypeIdentifier, voucher encoding.Encodable) (datatransfer2.Request, error) { + vbytes, err := encoding.Encode(voucher) + if err != nil { + return nil, xerrors.Errorf("Creating request: %w", err) + } + return &TransferRequest1_1{ + Type: uint64(types.VoucherMessage), + Vouch: &cborgen.Deferred{Raw: vbytes}, + VTyp: vtype, + XferID: uint64(id), + }, nil +} + +// RestartResponse builds a new Data Transfer response +func RestartResponse(id datatransfer2.TransferID, accepted bool, isPaused bool, voucherResultType datatransfer2.TypeIdentifier, voucherResult encoding.Encodable) (datatransfer2.Response, error) { + vbytes, err := encoding.Encode(voucherResult) + if err != nil { + return nil, xerrors.Errorf("Creating request: %w", err) + } + return &TransferResponse1_1{ + Acpt: accepted, + Type: uint64(types.RestartMessage), + Paus: isPaused, + XferID: uint64(id), + VTyp: voucherResultType, + VRes: &cborgen.Deferred{Raw: vbytes}, + }, nil +} + +// NewResponse builds a new Data Transfer response +func NewResponse(id datatransfer2.TransferID, accepted bool, isPaused bool, voucherResultType datatransfer2.TypeIdentifier, voucherResult encoding.Encodable) (datatransfer2.Response, error) { + vbytes, err := encoding.Encode(voucherResult) + if err != nil { + return nil, xerrors.Errorf("Creating request: %w", err) + } + return &TransferResponse1_1{ + Acpt: accepted, + Type: uint64(types.NewMessage), + Paus: isPaused, + XferID: uint64(id), + VTyp: voucherResultType, + VRes: &cborgen.Deferred{Raw: vbytes}, + }, nil +} + +// VoucherResultResponse builds a new response for a voucher result +func VoucherResultResponse(id datatransfer2.TransferID, accepted bool, isPaused bool, voucherResultType datatransfer2.TypeIdentifier, voucherResult encoding.Encodable) (datatransfer2.Response, error) { + vbytes, err := encoding.Encode(voucherResult) + if err != nil { + return nil, xerrors.Errorf("Creating request: %w", err) + } + return &TransferResponse1_1{ + Acpt: accepted, + Type: uint64(types.VoucherResultMessage), + Paus: isPaused, + XferID: uint64(id), + VTyp: voucherResultType, + VRes: &cborgen.Deferred{Raw: vbytes}, + }, nil +} + +// UpdateResponse returns a new update response +func UpdateResponse(id datatransfer2.TransferID, isPaused bool) datatransfer2.Response { + return &TransferResponse1_1{ + Type: uint64(types.UpdateMessage), + Paus: isPaused, + XferID: uint64(id), + } +} + +// CancelResponse makes a new cancel response message +func CancelResponse(id datatransfer2.TransferID) datatransfer2.Response { + return &TransferResponse1_1{ + Type: uint64(types.CancelMessage), + XferID: uint64(id), + } +} + +// CompleteResponse returns a new complete response message +func CompleteResponse(id datatransfer2.TransferID, isAccepted bool, isPaused bool, voucherResultType datatransfer2.TypeIdentifier, voucherResult encoding.Encodable) (datatransfer2.Response, error) { + vbytes, err := encoding.Encode(voucherResult) + if err != nil { + return nil, xerrors.Errorf("Creating request: %w", err) + } + return &TransferResponse1_1{ + Type: uint64(types.CompleteMessage), + Acpt: isAccepted, + Paus: isPaused, + VTyp: voucherResultType, + VRes: &cborgen.Deferred{Raw: vbytes}, + XferID: uint64(id), + }, nil +} + +// FromNet can read a network stream to deserialize a GraphSyncMessage +func FromNet(r io.Reader) (datatransfer2.Message, error) { + tresp := TransferMessage1_1{} + err := tresp.UnmarshalCBOR(r) + if err != nil { + return nil, err + } + + if (tresp.IsRequest() && tresp.Request == nil) || (!tresp.IsRequest() && tresp.Response == nil) { + return nil, xerrors.Errorf("invalid/malformed message") + } + + if tresp.IsRequest() { + return tresp.Request, nil + } + return tresp.Response, nil +} + +// FromNet can read a network stream to deserialize a GraphSyncMessage +func FromIPLD(nd datamodel.Node) (datatransfer2.Message, error) { + buf := new(bytes.Buffer) + err := dagcbor.Encode(nd, buf) + if err != nil { + return nil, err + } + return FromNet(buf) +} diff --git a/datatransfer/message/message1_1/transfer_message.go b/datatransfer/message/message1_1/transfer_message.go new file mode 100644 index 000000000..42c7493af --- /dev/null +++ b/datatransfer/message/message1_1/transfer_message.go @@ -0,0 +1,58 @@ +package message1_1 + +import ( + "bytes" + "io" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/ipld/go-ipld-prime/codec/dagcbor" + "github.com/ipld/go-ipld-prime/datamodel" + basicnode "github.com/ipld/go-ipld-prime/node/basic" +) + +//go:generate cbor-gen-for --map-encoding TransferMessage1_1 + +// transferMessage1_1 is the transfer message for the 1.1 Data Transfer Protocol. +type TransferMessage1_1 struct { + IsRq bool + + Request *TransferRequest1_1 + Response *TransferResponse1_1 +} + +// ========= datatransfer.Message interface + +// IsRequest returns true if this message is a data request +func (tm *TransferMessage1_1) IsRequest() bool { + return tm.IsRq +} + +// TransferID returns the TransferID of this message +func (tm *TransferMessage1_1) TransferID() datatransfer.TransferID { + if tm.IsRequest() { + return tm.Request.TransferID() + } + return tm.Response.TransferID() +} + +// ToNet serializes a transfer message type. It is simply a wrapper for MarshalCBOR, to provide +// symmetry with FromNet +func (tm *TransferMessage1_1) ToIPLD() (datamodel.Node, error) { + buf := new(bytes.Buffer) + err := tm.ToNet(buf) + if err != nil { + return nil, err + } + nb := basicnode.Prototype.Any.NewBuilder() + err = dagcbor.Decode(nb, buf) + if err != nil { + return nil, err + } + return nb.Build(), nil +} + +// ToNet serializes a transfer message type. It is simply a wrapper for MarshalCBOR, to provide +// symmetry with FromNet +func (tm *TransferMessage1_1) ToNet(w io.Writer) error { + return tm.MarshalCBOR(w) +} diff --git a/datatransfer/message/message1_1/transfer_message_cbor_gen.go b/datatransfer/message/message1_1/transfer_message_cbor_gen.go new file mode 100644 index 000000000..13d462b29 --- /dev/null +++ b/datatransfer/message/message1_1/transfer_message_cbor_gen.go @@ -0,0 +1,187 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package message1_1 + +import ( + "fmt" + "io" + "math" + "sort" + + cid "github.com/ipfs/go-cid" + cbg "github.com/whyrusleeping/cbor-gen" + xerrors "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +func (t *TransferMessage1_1) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{163}); err != nil { + return err + } + + // t.IsRq (bool) (bool) + if len("IsRq") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"IsRq\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("IsRq"))); err != nil { + return err + } + if _, err := cw.WriteString(string("IsRq")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.IsRq); err != nil { + return err + } + + // t.Request (message1_1.TransferRequest1_1) (struct) + if len("Request") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Request\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Request"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Request")); err != nil { + return err + } + + if err := t.Request.MarshalCBOR(cw); err != nil { + return err + } + + // t.Response (message1_1.TransferResponse1_1) (struct) + if len("Response") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Response\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Response"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Response")); err != nil { + return err + } + + if err := t.Response.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *TransferMessage1_1) UnmarshalCBOR(r io.Reader) (err error) { + *t = TransferMessage1_1{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("TransferMessage1_1: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.IsRq (bool) (bool) + case "IsRq": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.IsRq = false + case 21: + t.IsRq = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.Request (message1_1.TransferRequest1_1) (struct) + case "Request": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Request = new(TransferRequest1_1) + if err := t.Request.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Request pointer: %w", err) + } + } + + } + // t.Response (message1_1.TransferResponse1_1) (struct) + case "Response": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Response = new(TransferResponse1_1) + if err := t.Response.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Response pointer: %w", err) + } + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} diff --git a/datatransfer/message/message1_1/transfer_request.go b/datatransfer/message/message1_1/transfer_request.go new file mode 100644 index 000000000..6ea55a775 --- /dev/null +++ b/datatransfer/message/message1_1/transfer_request.go @@ -0,0 +1,165 @@ +package message1_1 + +import ( + "bytes" + "io" + + datatransfer2 "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/encoding" + "github.com/filecoin-project/boost/datatransfer/message/types" + "github.com/ipfs/go-cid" + "github.com/ipld/go-ipld-prime" + "github.com/ipld/go-ipld-prime/codec/dagcbor" + "github.com/ipld/go-ipld-prime/datamodel" + basicnode "github.com/ipld/go-ipld-prime/node/basic" + "github.com/libp2p/go-libp2p/core/protocol" + cbg "github.com/whyrusleeping/cbor-gen" + "golang.org/x/xerrors" +) + +//go:generate cbor-gen-for --map-encoding TransferRequest1_1 + +// TransferRequest1_1 is a struct for the 1.1 Data Transfer Protocol that fulfills the datatransfer.Request interface. +// its members are exported to be used by cbor-gen +type TransferRequest1_1 struct { + BCid *cid.Cid + Type uint64 + Paus bool + Part bool + Pull bool + Stor *cbg.Deferred + Vouch *cbg.Deferred + VTyp datatransfer2.TypeIdentifier + XferID uint64 + + RestartChannel datatransfer2.ChannelID +} + +func (trq *TransferRequest1_1) MessageForProtocol(targetProtocol protocol.ID) (datatransfer2.Message, error) { + switch targetProtocol { + case datatransfer2.ProtocolDataTransfer1_2: + return trq, nil + default: + return nil, xerrors.Errorf("protocol not supported") + } +} + +// IsRequest always returns true in this case because this is a transfer request +func (trq *TransferRequest1_1) IsRequest() bool { + return true +} + +func (trq *TransferRequest1_1) IsRestart() bool { + return trq.Type == uint64(types.RestartMessage) +} + +func (trq *TransferRequest1_1) IsRestartExistingChannelRequest() bool { + return trq.Type == uint64(types.RestartExistingChannelRequestMessage) +} + +func (trq *TransferRequest1_1) RestartChannelId() (datatransfer2.ChannelID, error) { + if !trq.IsRestartExistingChannelRequest() { + return datatransfer2.ChannelID{}, xerrors.New("not a restart request") + } + return trq.RestartChannel, nil +} + +func (trq *TransferRequest1_1) IsNew() bool { + return trq.Type == uint64(types.NewMessage) +} + +func (trq *TransferRequest1_1) IsUpdate() bool { + return trq.Type == uint64(types.UpdateMessage) +} + +func (trq *TransferRequest1_1) IsVoucher() bool { + return trq.Type == uint64(types.VoucherMessage) || trq.Type == uint64(types.NewMessage) +} + +func (trq *TransferRequest1_1) IsPaused() bool { + return trq.Paus +} + +func (trq *TransferRequest1_1) TransferID() datatransfer2.TransferID { + return datatransfer2.TransferID(trq.XferID) +} + +// ========= datatransfer.Request interface +// IsPull returns true if this is a data pull request +func (trq *TransferRequest1_1) IsPull() bool { + return trq.Pull +} + +// VoucherType returns the Voucher ID +func (trq *TransferRequest1_1) VoucherType() datatransfer2.TypeIdentifier { + return trq.VTyp +} + +// Voucher returns the Voucher bytes +func (trq *TransferRequest1_1) Voucher(decoder encoding.Decoder) (encoding.Encodable, error) { + if trq.Vouch == nil { + return nil, xerrors.New("No voucher present to read") + } + return decoder.DecodeFromCbor(trq.Vouch.Raw) +} + +func (trq *TransferRequest1_1) EmptyVoucher() bool { + return trq.VTyp == datatransfer2.EmptyTypeIdentifier +} + +// BaseCid returns the Base CID +func (trq *TransferRequest1_1) BaseCid() cid.Cid { + if trq.BCid == nil { + return cid.Undef + } + return *trq.BCid +} + +// Selector returns the message Selector bytes +func (trq *TransferRequest1_1) Selector() (ipld.Node, error) { + if trq.Stor == nil { + return nil, xerrors.New("No selector present to read") + } + builder := basicnode.Prototype.Any.NewBuilder() + reader := bytes.NewReader(trq.Stor.Raw) + err := dagcbor.Decode(builder, reader) + if err != nil { + return nil, xerrors.Errorf("Error decoding selector: %w", err) + } + return builder.Build(), nil +} + +// IsCancel returns true if this is a cancel request +func (trq *TransferRequest1_1) IsCancel() bool { + return trq.Type == uint64(types.CancelMessage) +} + +// IsPartial returns true if this is a partial request +func (trq *TransferRequest1_1) IsPartial() bool { + return trq.Part +} + +func (trq *TransferRequest1_1) ToIPLD() (datamodel.Node, error) { + buf := new(bytes.Buffer) + err := trq.ToNet(buf) + if err != nil { + return nil, err + } + nb := basicnode.Prototype.Any.NewBuilder() + err = dagcbor.Decode(nb, buf) + if err != nil { + return nil, err + } + return nb.Build(), nil +} + +// ToNet serializes a transfer request. It's a wrapper for MarshalCBOR to provide +// symmetry with FromNet +func (trq *TransferRequest1_1) ToNet(w io.Writer) error { + msg := TransferMessage1_1{ + IsRq: true, + Request: trq, + Response: nil, + } + return msg.MarshalCBOR(w) +} diff --git a/datatransfer/message/message1_1/transfer_request_cbor_gen.go b/datatransfer/message/message1_1/transfer_request_cbor_gen.go new file mode 100644 index 000000000..3e3214ceb --- /dev/null +++ b/datatransfer/message/message1_1/transfer_request_cbor_gen.go @@ -0,0 +1,405 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package message1_1 + +import ( + "fmt" + "io" + "math" + "sort" + + datatransfer "github.com/filecoin-project/boost/datatransfer" + cid "github.com/ipfs/go-cid" + cbg "github.com/whyrusleeping/cbor-gen" + xerrors "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +func (t *TransferRequest1_1) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{170}); err != nil { + return err + } + + // t.BCid (cid.Cid) (struct) + if len("BCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"BCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("BCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("BCid")); err != nil { + return err + } + + if t.BCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.BCid); err != nil { + return xerrors.Errorf("failed to write cid field t.BCid: %w", err) + } + } + + // t.Part (bool) (bool) + if len("Part") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Part\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Part"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Part")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.Part); err != nil { + return err + } + + // t.Paus (bool) (bool) + if len("Paus") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Paus\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Paus"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Paus")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.Paus); err != nil { + return err + } + + // t.Pull (bool) (bool) + if len("Pull") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Pull\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Pull"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Pull")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.Pull); err != nil { + return err + } + + // t.Stor (typegen.Deferred) (struct) + if len("Stor") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Stor\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Stor"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Stor")); err != nil { + return err + } + + if err := t.Stor.MarshalCBOR(cw); err != nil { + return err + } + + // t.Type (uint64) (uint64) + if len("Type") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Type\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Type"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Type")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Type)); err != nil { + return err + } + + // t.VTyp (datatransfer.TypeIdentifier) (string) + if len("VTyp") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"VTyp\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("VTyp"))); err != nil { + return err + } + if _, err := cw.WriteString(string("VTyp")); err != nil { + return err + } + + if len(t.VTyp) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.VTyp was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.VTyp))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.VTyp)); err != nil { + return err + } + + // t.Vouch (typegen.Deferred) (struct) + if len("Vouch") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Vouch\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Vouch"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Vouch")); err != nil { + return err + } + + if err := t.Vouch.MarshalCBOR(cw); err != nil { + return err + } + + // t.XferID (uint64) (uint64) + if len("XferID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"XferID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("XferID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("XferID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.XferID)); err != nil { + return err + } + + // t.RestartChannel (datatransfer.ChannelID) (struct) + if len("RestartChannel") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"RestartChannel\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("RestartChannel"))); err != nil { + return err + } + if _, err := cw.WriteString(string("RestartChannel")); err != nil { + return err + } + + if err := t.RestartChannel.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *TransferRequest1_1) UnmarshalCBOR(r io.Reader) (err error) { + *t = TransferRequest1_1{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("TransferRequest1_1: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.BCid (cid.Cid) (struct) + case "BCid": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.BCid: %w", err) + } + + t.BCid = &c + } + + } + // t.Part (bool) (bool) + case "Part": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.Part = false + case 21: + t.Part = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.Paus (bool) (bool) + case "Paus": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.Paus = false + case 21: + t.Paus = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.Pull (bool) (bool) + case "Pull": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.Pull = false + case 21: + t.Pull = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.Stor (typegen.Deferred) (struct) + case "Stor": + + { + + t.Stor = new(cbg.Deferred) + + if err := t.Stor.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("failed to read deferred field: %w", err) + } + } + // t.Type (uint64) (uint64) + case "Type": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Type = uint64(extra) + + } + // t.VTyp (datatransfer.TypeIdentifier) (string) + case "VTyp": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.VTyp = datatransfer.TypeIdentifier(sval) + } + // t.Vouch (typegen.Deferred) (struct) + case "Vouch": + + { + + t.Vouch = new(cbg.Deferred) + + if err := t.Vouch.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("failed to read deferred field: %w", err) + } + } + // t.XferID (uint64) (uint64) + case "XferID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.XferID = uint64(extra) + + } + // t.RestartChannel (datatransfer.ChannelID) (struct) + case "RestartChannel": + + { + + if err := t.RestartChannel.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.RestartChannel: %w", err) + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} diff --git a/datatransfer/message/message1_1/transfer_response.go b/datatransfer/message/message1_1/transfer_response.go new file mode 100644 index 000000000..d62a082c8 --- /dev/null +++ b/datatransfer/message/message1_1/transfer_response.go @@ -0,0 +1,126 @@ +package message1_1 + +import ( + "bytes" + "io" + + datatransfer2 "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/encoding" + "github.com/filecoin-project/boost/datatransfer/message/types" + "github.com/ipld/go-ipld-prime/codec/dagcbor" + "github.com/ipld/go-ipld-prime/datamodel" + basicnode "github.com/ipld/go-ipld-prime/node/basic" + "github.com/libp2p/go-libp2p/core/protocol" + cbg "github.com/whyrusleeping/cbor-gen" + "golang.org/x/xerrors" +) + +//go:generate cbor-gen-for --map-encoding TransferResponse1_1 + +// TransferResponse1_1 is a private struct that satisfies the datatransfer.Response interface +// It is the response message for the Data Transfer 1.1 and 1.2 Protocol. +type TransferResponse1_1 struct { + Type uint64 + Acpt bool + Paus bool + XferID uint64 + VRes *cbg.Deferred + VTyp datatransfer2.TypeIdentifier +} + +func (trsp *TransferResponse1_1) TransferID() datatransfer2.TransferID { + return datatransfer2.TransferID(trsp.XferID) +} + +// IsRequest always returns false in this case because this is a transfer response +func (trsp *TransferResponse1_1) IsRequest() bool { + return false +} + +// IsNew returns true if this is the first response sent +func (trsp *TransferResponse1_1) IsNew() bool { + return trsp.Type == uint64(types.NewMessage) +} + +// IsUpdate returns true if this response is an update +func (trsp *TransferResponse1_1) IsUpdate() bool { + return trsp.Type == uint64(types.UpdateMessage) +} + +// IsPaused returns true if the responder is paused +func (trsp *TransferResponse1_1) IsPaused() bool { + return trsp.Paus +} + +// IsCancel returns true if the responder has cancelled this response +func (trsp *TransferResponse1_1) IsCancel() bool { + return trsp.Type == uint64(types.CancelMessage) +} + +// IsComplete returns true if the responder has completed this response +func (trsp *TransferResponse1_1) IsComplete() bool { + return trsp.Type == uint64(types.CompleteMessage) +} + +func (trsp *TransferResponse1_1) IsVoucherResult() bool { + return trsp.Type == uint64(types.VoucherResultMessage) || trsp.Type == uint64(types.NewMessage) || trsp.Type == uint64(types.CompleteMessage) || + trsp.Type == uint64(types.RestartMessage) +} + +// Accepted returns true if the request is accepted in the response +func (trsp *TransferResponse1_1) Accepted() bool { + return trsp.Acpt +} + +func (trsp *TransferResponse1_1) VoucherResultType() datatransfer2.TypeIdentifier { + return trsp.VTyp +} + +func (trsp *TransferResponse1_1) VoucherResult(decoder encoding.Decoder) (encoding.Encodable, error) { + if trsp.VRes == nil { + return nil, xerrors.New("No voucher present to read") + } + return decoder.DecodeFromCbor(trsp.VRes.Raw) +} + +func (trq *TransferResponse1_1) IsRestart() bool { + return trq.Type == uint64(types.RestartMessage) +} + +func (trsp *TransferResponse1_1) EmptyVoucherResult() bool { + return trsp.VTyp == datatransfer2.EmptyTypeIdentifier +} + +func (trsp *TransferResponse1_1) MessageForProtocol(targetProtocol protocol.ID) (datatransfer2.Message, error) { + switch targetProtocol { + case datatransfer2.ProtocolDataTransfer1_2: + return trsp, nil + default: + return nil, xerrors.Errorf("protocol %s not supported", targetProtocol) + } +} + +func (trsp *TransferResponse1_1) ToIPLD() (datamodel.Node, error) { + buf := new(bytes.Buffer) + err := trsp.ToNet(buf) + if err != nil { + return nil, err + } + nb := basicnode.Prototype.Any.NewBuilder() + err = dagcbor.Decode(nb, buf) + if err != nil { + return nil, err + } + return nb.Build(), nil +} + +// ToNet serializes a transfer response. It's a wrapper for MarshalCBOR to provide +// symmetry with FromNet +func (trsp *TransferResponse1_1) ToNet(w io.Writer) error { + msg := TransferMessage1_1{ + IsRq: false, + Request: nil, + Response: trsp, + } + return msg.MarshalCBOR(w) +} diff --git a/datatransfer/message/message1_1/transfer_response_cbor_gen.go b/datatransfer/message/message1_1/transfer_response_cbor_gen.go new file mode 100644 index 000000000..35ec7f90d --- /dev/null +++ b/datatransfer/message/message1_1/transfer_response_cbor_gen.go @@ -0,0 +1,274 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package message1_1 + +import ( + "fmt" + "io" + "math" + "sort" + + datatransfer "github.com/filecoin-project/boost/datatransfer" + cid "github.com/ipfs/go-cid" + cbg "github.com/whyrusleeping/cbor-gen" + xerrors "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +func (t *TransferResponse1_1) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{166}); err != nil { + return err + } + + // t.Acpt (bool) (bool) + if len("Acpt") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Acpt\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Acpt"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Acpt")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.Acpt); err != nil { + return err + } + + // t.Paus (bool) (bool) + if len("Paus") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Paus\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Paus"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Paus")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.Paus); err != nil { + return err + } + + // t.Type (uint64) (uint64) + if len("Type") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Type\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Type"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Type")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Type)); err != nil { + return err + } + + // t.VRes (typegen.Deferred) (struct) + if len("VRes") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"VRes\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("VRes"))); err != nil { + return err + } + if _, err := cw.WriteString(string("VRes")); err != nil { + return err + } + + if err := t.VRes.MarshalCBOR(cw); err != nil { + return err + } + + // t.VTyp (datatransfer.TypeIdentifier) (string) + if len("VTyp") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"VTyp\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("VTyp"))); err != nil { + return err + } + if _, err := cw.WriteString(string("VTyp")); err != nil { + return err + } + + if len(t.VTyp) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.VTyp was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.VTyp))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.VTyp)); err != nil { + return err + } + + // t.XferID (uint64) (uint64) + if len("XferID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"XferID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("XferID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("XferID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.XferID)); err != nil { + return err + } + + return nil +} + +func (t *TransferResponse1_1) UnmarshalCBOR(r io.Reader) (err error) { + *t = TransferResponse1_1{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("TransferResponse1_1: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Acpt (bool) (bool) + case "Acpt": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.Acpt = false + case 21: + t.Acpt = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.Paus (bool) (bool) + case "Paus": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.Paus = false + case 21: + t.Paus = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.Type (uint64) (uint64) + case "Type": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Type = uint64(extra) + + } + // t.VRes (typegen.Deferred) (struct) + case "VRes": + + { + + t.VRes = new(cbg.Deferred) + + if err := t.VRes.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("failed to read deferred field: %w", err) + } + } + // t.VTyp (datatransfer.TypeIdentifier) (string) + case "VTyp": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.VTyp = datatransfer.TypeIdentifier(sval) + } + // t.XferID (uint64) (uint64) + case "XferID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.XferID = uint64(extra) + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} diff --git a/datatransfer/message/message1_1prime/message.go b/datatransfer/message/message1_1prime/message.go new file mode 100644 index 000000000..d4d32897f --- /dev/null +++ b/datatransfer/message/message1_1prime/message.go @@ -0,0 +1,206 @@ +package message1_1 + +import ( + "io" + + datatransfer2 "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/encoding" + "github.com/filecoin-project/boost/datatransfer/message/types" + "github.com/ipfs/go-cid" + "github.com/ipld/go-ipld-prime" + "github.com/ipld/go-ipld-prime/codec/dagcbor" + "github.com/ipld/go-ipld-prime/datamodel" + "github.com/ipld/go-ipld-prime/node/bindnode" + "github.com/ipld/go-ipld-prime/schema" + "golang.org/x/xerrors" +) + +// NewRequest generates a new request for the data transfer protocol +func NewRequest(id datatransfer2.TransferID, isRestart bool, isPull bool, vtype datatransfer2.TypeIdentifier, voucher encoding.Encodable, baseCid cid.Cid, selector ipld.Node) (datatransfer2.Request, error) { + vnode, err := encoding.EncodeToNode(voucher) + if err != nil { + return nil, xerrors.Errorf("Creating request: %w", err) + } + + if baseCid == cid.Undef { + return nil, xerrors.Errorf("base CID must be defined") + } + + var typ uint64 + if isRestart { + typ = uint64(types.RestartMessage) + } else { + typ = uint64(types.NewMessage) + } + + return &TransferRequest1_1{ + MessageType: typ, + Pull: isPull, + VoucherPtr: &vnode, + SelectorPtr: &selector, + BaseCidPtr: &baseCid, + VoucherTypeIdentifier: vtype, + TransferId: uint64(id), + }, nil +} + +// RestartExistingChannelRequest creates a request to ask the other side to restart an existing channel +func RestartExistingChannelRequest(channelId datatransfer2.ChannelID) datatransfer2.Request { + return &TransferRequest1_1{ + MessageType: uint64(types.RestartExistingChannelRequestMessage), + RestartChannel: channelId, + } +} + +// CancelRequest request generates a request to cancel an in progress request +func CancelRequest(id datatransfer2.TransferID) datatransfer2.Request { + return &TransferRequest1_1{ + MessageType: uint64(types.CancelMessage), + TransferId: uint64(id), + } +} + +// UpdateRequest generates a new request update +func UpdateRequest(id datatransfer2.TransferID, isPaused bool) datatransfer2.Request { + return &TransferRequest1_1{ + MessageType: uint64(types.UpdateMessage), + Pause: isPaused, + TransferId: uint64(id), + } +} + +// VoucherRequest generates a new request for the data transfer protocol +func VoucherRequest(id datatransfer2.TransferID, vtype datatransfer2.TypeIdentifier, voucher encoding.Encodable) (datatransfer2.Request, error) { + vnode, err := encoding.EncodeToNode(voucher) + if err != nil { + return nil, xerrors.Errorf("Creating request: %w", err) + } + return &TransferRequest1_1{ + MessageType: uint64(types.VoucherMessage), + VoucherPtr: &vnode, + VoucherTypeIdentifier: vtype, + TransferId: uint64(id), + }, nil +} + +// RestartResponse builds a new Data Transfer response +func RestartResponse(id datatransfer2.TransferID, accepted bool, isPaused bool, voucherResultType datatransfer2.TypeIdentifier, voucherResult encoding.Encodable) (datatransfer2.Response, error) { + vnode, err := encoding.EncodeToNode(voucherResult) + if err != nil { + return nil, xerrors.Errorf("Creating request: %w", err) + } + return &TransferResponse1_1{ + RequestAccepted: accepted, + MessageType: uint64(types.RestartMessage), + Paused: isPaused, + TransferId: uint64(id), + VoucherTypeIdentifier: voucherResultType, + VoucherResultPtr: &vnode, + }, nil +} + +// NewResponse builds a new Data Transfer response +func NewResponse(id datatransfer2.TransferID, accepted bool, isPaused bool, voucherResultType datatransfer2.TypeIdentifier, voucherResult encoding.Encodable) (datatransfer2.Response, error) { + vnode, err := encoding.EncodeToNode(voucherResult) + if err != nil { + return nil, xerrors.Errorf("Creating request: %w", err) + } + return &TransferResponse1_1{ + RequestAccepted: accepted, + MessageType: uint64(types.NewMessage), + Paused: isPaused, + TransferId: uint64(id), + VoucherTypeIdentifier: voucherResultType, + VoucherResultPtr: &vnode, + }, nil +} + +// VoucherResultResponse builds a new response for a voucher result +func VoucherResultResponse(id datatransfer2.TransferID, accepted bool, isPaused bool, voucherResultType datatransfer2.TypeIdentifier, voucherResult encoding.Encodable) (datatransfer2.Response, error) { + vnode, err := encoding.EncodeToNode(voucherResult) + if err != nil { + return nil, xerrors.Errorf("Creating request: %w", err) + } + return &TransferResponse1_1{ + RequestAccepted: accepted, + MessageType: uint64(types.VoucherResultMessage), + Paused: isPaused, + TransferId: uint64(id), + VoucherTypeIdentifier: voucherResultType, + VoucherResultPtr: &vnode, + }, nil +} + +// UpdateResponse returns a new update response +func UpdateResponse(id datatransfer2.TransferID, isPaused bool) datatransfer2.Response { + return &TransferResponse1_1{ + MessageType: uint64(types.UpdateMessage), + Paused: isPaused, + TransferId: uint64(id), + } +} + +// CancelResponse makes a new cancel response message +func CancelResponse(id datatransfer2.TransferID) datatransfer2.Response { + return &TransferResponse1_1{ + MessageType: uint64(types.CancelMessage), + TransferId: uint64(id), + } +} + +// CompleteResponse returns a new complete response message +func CompleteResponse(id datatransfer2.TransferID, isAccepted bool, isPaused bool, voucherResultType datatransfer2.TypeIdentifier, voucherResult encoding.Encodable) (datatransfer2.Response, error) { + vnode, err := encoding.EncodeToNode(voucherResult) + if err != nil { + return nil, xerrors.Errorf("Creating request: %w", err) + } + return &TransferResponse1_1{ + MessageType: uint64(types.CompleteMessage), + RequestAccepted: isAccepted, + Paused: isPaused, + VoucherTypeIdentifier: voucherResultType, + VoucherResultPtr: &vnode, + TransferId: uint64(id), + }, nil +} + +// FromNet can read a network stream to deserialize a GraphSyncMessage +func FromNet(r io.Reader) (datatransfer2.Message, error) { + builder := Prototype.TransferMessage.Representation().NewBuilder() + err := dagcbor.Decode(builder, r) + if err != nil { + return nil, err + } + node := builder.Build() + tresp := bindnode.Unwrap(node).(*TransferMessage1_1) + + if (tresp.IsRequest && tresp.Request == nil) || (!tresp.IsRequest && tresp.Response == nil) { + return nil, xerrors.Errorf("invalid/malformed message") + } + + if tresp.IsRequest { + return tresp.Request, nil + } + return tresp.Response, nil +} + +// FromNet can read a network stream to deserialize a GraphSyncMessage +func FromIPLD(node datamodel.Node) (datatransfer2.Message, error) { + if tn, ok := node.(schema.TypedNode); ok { // shouldn't need this if from Graphsync + node = tn.Representation() + } + builder := Prototype.TransferMessage.Representation().NewBuilder() + err := builder.AssignNode(node) + if err != nil { + return nil, err + } + tresp := bindnode.Unwrap(builder.Build()).(*TransferMessage1_1) + if (tresp.IsRequest && tresp.Request == nil) || (!tresp.IsRequest && tresp.Response == nil) { + return nil, xerrors.Errorf("invalid/malformed message") + } + + if tresp.IsRequest { + return tresp.Request, nil + } + return tresp.Response, nil +} diff --git a/datatransfer/message/message1_1prime/schema.go b/datatransfer/message/message1_1prime/schema.go new file mode 100644 index 000000000..c779b1fc9 --- /dev/null +++ b/datatransfer/message/message1_1prime/schema.go @@ -0,0 +1,29 @@ +package message1_1 + +import ( + _ "embed" + + "github.com/ipld/go-ipld-prime" + "github.com/ipld/go-ipld-prime/node/bindnode" + "github.com/ipld/go-ipld-prime/schema" +) + +//go:embed schema.ipldsch +var embedSchema []byte + +var Prototype struct { + TransferMessage schema.TypedPrototype + TransferRequest schema.TypedPrototype + TransferResponse schema.TypedPrototype +} + +func init() { + ts, err := ipld.LoadSchemaBytes(embedSchema) + if err != nil { + panic(err) + } + + Prototype.TransferMessage = bindnode.Prototype((*TransferMessage1_1)(nil), ts.TypeByName("TransferMessage")) + Prototype.TransferRequest = bindnode.Prototype((*TransferRequest1_1)(nil), ts.TypeByName("TransferRequest")) + Prototype.TransferResponse = bindnode.Prototype((*TransferResponse1_1)(nil), ts.TypeByName("TransferResponse")) +} diff --git a/datatransfer/message/message1_1prime/schema.ipldsch b/datatransfer/message/message1_1prime/schema.ipldsch new file mode 100644 index 000000000..714135141 --- /dev/null +++ b/datatransfer/message/message1_1prime/schema.ipldsch @@ -0,0 +1,37 @@ +type PeerID string # peer.ID, really should be bytes (this is non-utf8) but is string for backward compat +type TransferID int +type TypeIdentifier string + +type ChannelID struct { + Initiator PeerID + Responder PeerID + ID TransferID +} representation tuple + +type TransferRequest struct { + BaseCidPtr nullable Link (rename "BCid") + MessageType Int (rename "Type") + Pause Bool (rename "Paus") + Partial Bool (rename "Part") + Pull Bool (rename "Pull") + SelectorPtr nullable Any (rename "Stor") + VoucherPtr nullable Any (rename "Vouch") + VoucherTypeIdentifier TypeIdentifier (rename "VTyp") + TransferId Int (rename "XferID") + RestartChannel ChannelID +} + +type TransferResponse struct { + MessageType Int (rename "Type") + RequestAccepted Bool (rename "Acpt") + Paused Bool (rename "Paus") + TransferId Int (rename "XferID") + VoucherResultPtr nullable Any (rename "VRes") + VoucherTypeIdentifier TypeIdentifier (rename "VTyp") +} + +type TransferMessage struct { + IsRequest Bool (rename "IsRq") + Request nullable TransferRequest + Response nullable TransferResponse +} diff --git a/datatransfer/message/message1_1prime/transfer_message.go b/datatransfer/message/message1_1prime/transfer_message.go new file mode 100644 index 000000000..f66b9f2d3 --- /dev/null +++ b/datatransfer/message/message1_1prime/transfer_message.go @@ -0,0 +1,43 @@ +package message1_1 + +import ( + "io" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/ipld/go-ipld-prime/codec/dagcbor" + "github.com/ipld/go-ipld-prime/datamodel" + "github.com/ipld/go-ipld-prime/node/bindnode" + "github.com/ipld/go-ipld-prime/schema" +) + +// TransferMessage1_1 is the transfer message for the 1.1 Data Transfer Protocol. +type TransferMessage1_1 struct { + IsRequest bool + + Request *TransferRequest1_1 + Response *TransferResponse1_1 +} + +// ========= datatransfer.Message interface + +// TransferID returns the TransferID of this message +func (tm *TransferMessage1_1) TransferID() datatransfer.TransferID { + if tm.IsRequest { + return tm.Request.TransferID() + } + return tm.Response.TransferID() +} + +func (tm *TransferMessage1_1) toIPLD() schema.TypedNode { + return bindnode.Wrap(tm, Prototype.TransferMessage.Type()) +} + +// ToNet serializes a transfer message type. +func (tm *TransferMessage1_1) ToIPLD() (datamodel.Node, error) { + return tm.toIPLD().Representation(), nil +} + +// ToNet serializes a transfer message type. +func (tm *TransferMessage1_1) ToNet(w io.Writer) error { + return dagcbor.Encode(tm.toIPLD().Representation(), w) +} diff --git a/datatransfer/message/message1_1prime/transfer_request.go b/datatransfer/message/message1_1prime/transfer_request.go new file mode 100644 index 000000000..7675a6b37 --- /dev/null +++ b/datatransfer/message/message1_1prime/transfer_request.go @@ -0,0 +1,146 @@ +package message1_1 + +import ( + "io" + + datatransfer2 "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/encoding" + "github.com/filecoin-project/boost/datatransfer/message/types" + "github.com/ipfs/go-cid" + "github.com/ipld/go-ipld-prime/codec/dagcbor" + "github.com/ipld/go-ipld-prime/datamodel" + "github.com/ipld/go-ipld-prime/schema" + "github.com/libp2p/go-libp2p/core/protocol" + "golang.org/x/xerrors" +) + +// TransferRequest1_1 is a struct for the 1.1 Data Transfer Protocol that fulfills the datatransfer.Request interface. +// its members are exported to be used by cbor-gen +type TransferRequest1_1 struct { + BaseCidPtr *cid.Cid + MessageType uint64 + Pause bool + Partial bool + Pull bool + SelectorPtr *datamodel.Node + VoucherPtr *datamodel.Node + VoucherTypeIdentifier datatransfer2.TypeIdentifier + TransferId uint64 + RestartChannel datatransfer2.ChannelID +} + +func (trq *TransferRequest1_1) MessageForProtocol(targetProtocol protocol.ID) (datatransfer2.Message, error) { + switch targetProtocol { + case datatransfer2.ProtocolDataTransfer1_2: + return trq, nil + default: + return nil, xerrors.Errorf("protocol not supported") + } +} + +// IsRequest always returns true in this case because this is a transfer request +func (trq *TransferRequest1_1) IsRequest() bool { + return true +} + +func (trq *TransferRequest1_1) IsRestart() bool { + return trq.MessageType == uint64(types.RestartMessage) +} + +func (trq *TransferRequest1_1) IsRestartExistingChannelRequest() bool { + return trq.MessageType == uint64(types.RestartExistingChannelRequestMessage) +} + +func (trq *TransferRequest1_1) RestartChannelId() (datatransfer2.ChannelID, error) { + if !trq.IsRestartExistingChannelRequest() { + return datatransfer2.ChannelID{}, xerrors.New("not a restart request") + } + return trq.RestartChannel, nil +} + +func (trq *TransferRequest1_1) IsNew() bool { + return trq.MessageType == uint64(types.NewMessage) +} + +func (trq *TransferRequest1_1) IsUpdate() bool { + return trq.MessageType == uint64(types.UpdateMessage) +} + +func (trq *TransferRequest1_1) IsVoucher() bool { + return trq.MessageType == uint64(types.VoucherMessage) || trq.MessageType == uint64(types.NewMessage) +} + +func (trq *TransferRequest1_1) IsPaused() bool { + return trq.Pause +} + +func (trq *TransferRequest1_1) TransferID() datatransfer2.TransferID { + return datatransfer2.TransferID(trq.TransferId) +} + +// ========= datatransfer.Request interface +// IsPull returns true if this is a data pull request +func (trq *TransferRequest1_1) IsPull() bool { + return trq.Pull +} + +// VoucherType returns the Voucher ID +func (trq *TransferRequest1_1) VoucherType() datatransfer2.TypeIdentifier { + return trq.VoucherTypeIdentifier +} + +// Voucher returns the Voucher bytes +func (trq *TransferRequest1_1) Voucher(decoder encoding.Decoder) (encoding.Encodable, error) { + if trq.VoucherPtr == nil { + return nil, xerrors.New("No voucher present to read") + } + return decoder.DecodeFromNode(*trq.VoucherPtr) +} + +func (trq *TransferRequest1_1) EmptyVoucher() bool { + return trq.VoucherTypeIdentifier == datatransfer2.EmptyTypeIdentifier +} + +// BaseCid returns the Base CID +func (trq *TransferRequest1_1) BaseCid() cid.Cid { + if trq.BaseCidPtr == nil { + return cid.Undef + } + return *trq.BaseCidPtr +} + +// Selector returns the message Selector bytes +func (trq *TransferRequest1_1) Selector() (datamodel.Node, error) { + if trq.SelectorPtr == nil { + return nil, xerrors.New("No selector present to read") + } + return *trq.SelectorPtr, nil +} + +// IsCancel returns true if this is a cancel request +func (trq *TransferRequest1_1) IsCancel() bool { + return trq.MessageType == uint64(types.CancelMessage) +} + +// IsPartial returns true if this is a partial request +func (trq *TransferRequest1_1) IsPartial() bool { + return trq.Partial +} + +func (trsp *TransferRequest1_1) toIPLD() schema.TypedNode { + msg := TransferMessage1_1{ + IsRequest: true, + Request: trsp, + Response: nil, + } + return msg.toIPLD() +} + +func (trq *TransferRequest1_1) ToIPLD() (datamodel.Node, error) { + return trq.toIPLD().Representation(), nil +} + +// ToNet serializes a transfer request. +func (trq *TransferRequest1_1) ToNet(w io.Writer) error { + return dagcbor.Encode(trq.toIPLD().Representation(), w) +} diff --git a/datatransfer/message/message1_1prime/transfer_response.go b/datatransfer/message/message1_1prime/transfer_response.go new file mode 100644 index 000000000..2b416b60a --- /dev/null +++ b/datatransfer/message/message1_1prime/transfer_response.go @@ -0,0 +1,115 @@ +package message1_1 + +import ( + "io" + + datatransfer2 "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/encoding" + "github.com/filecoin-project/boost/datatransfer/message/types" + "github.com/ipld/go-ipld-prime/codec/dagcbor" + "github.com/ipld/go-ipld-prime/datamodel" + "github.com/ipld/go-ipld-prime/schema" + "github.com/libp2p/go-libp2p/core/protocol" + "golang.org/x/xerrors" +) + +// TransferResponse1_1 is a private struct that satisfies the datatransfer.Response interface +// It is the response message for the Data Transfer 1.1 and 1.2 Protocol. +type TransferResponse1_1 struct { + MessageType uint64 + RequestAccepted bool + Paused bool + TransferId uint64 + VoucherResultPtr *datamodel.Node + VoucherTypeIdentifier datatransfer2.TypeIdentifier +} + +func (trsp *TransferResponse1_1) TransferID() datatransfer2.TransferID { + return datatransfer2.TransferID(trsp.TransferId) +} + +// IsRequest always returns false in this case because this is a transfer response +func (trsp *TransferResponse1_1) IsRequest() bool { + return false +} + +// IsNew returns true if this is the first response sent +func (trsp *TransferResponse1_1) IsNew() bool { + return trsp.MessageType == uint64(types.NewMessage) +} + +// IsUpdate returns true if this response is an update +func (trsp *TransferResponse1_1) IsUpdate() bool { + return trsp.MessageType == uint64(types.UpdateMessage) +} + +// IsPaused returns true if the responder is paused +func (trsp *TransferResponse1_1) IsPaused() bool { + return trsp.Paused +} + +// IsCancel returns true if the responder has cancelled this response +func (trsp *TransferResponse1_1) IsCancel() bool { + return trsp.MessageType == uint64(types.CancelMessage) +} + +// IsComplete returns true if the responder has completed this response +func (trsp *TransferResponse1_1) IsComplete() bool { + return trsp.MessageType == uint64(types.CompleteMessage) +} + +func (trsp *TransferResponse1_1) IsVoucherResult() bool { + return trsp.MessageType == uint64(types.VoucherResultMessage) || trsp.MessageType == uint64(types.NewMessage) || trsp.MessageType == uint64(types.CompleteMessage) || + trsp.MessageType == uint64(types.RestartMessage) +} + +// Accepted returns true if the request is accepted in the response +func (trsp *TransferResponse1_1) Accepted() bool { + return trsp.RequestAccepted +} + +func (trsp *TransferResponse1_1) VoucherResultType() datatransfer2.TypeIdentifier { + return trsp.VoucherTypeIdentifier +} + +func (trsp *TransferResponse1_1) VoucherResult(decoder encoding.Decoder) (encoding.Encodable, error) { + if trsp.VoucherResultPtr == nil { + return nil, xerrors.New("No voucher present to read") + } + return decoder.DecodeFromNode(*trsp.VoucherResultPtr) +} + +func (trq *TransferResponse1_1) IsRestart() bool { + return trq.MessageType == uint64(types.RestartMessage) +} + +func (trsp *TransferResponse1_1) EmptyVoucherResult() bool { + return trsp.VoucherTypeIdentifier == datatransfer2.EmptyTypeIdentifier +} + +func (trsp *TransferResponse1_1) MessageForProtocol(targetProtocol protocol.ID) (datatransfer2.Message, error) { + switch targetProtocol { + case datatransfer2.ProtocolDataTransfer1_2: + return trsp, nil + default: + return nil, xerrors.Errorf("protocol %s not supported", targetProtocol) + } +} + +func (trsp *TransferResponse1_1) toIPLD() schema.TypedNode { + msg := TransferMessage1_1{ + IsRequest: false, + Request: nil, + Response: trsp, + } + return msg.toIPLD() +} + +func (trsp *TransferResponse1_1) ToIPLD() (datamodel.Node, error) { + return trsp.toIPLD().Representation(), nil +} + +// ToNet serializes a transfer response. +func (trsp *TransferResponse1_1) ToNet(w io.Writer) error { + return dagcbor.Encode(trsp.toIPLD().Representation(), w) +} diff --git a/datatransfer/message/types/message_types.go b/datatransfer/message/types/message_types.go new file mode 100644 index 000000000..3144df0a2 --- /dev/null +++ b/datatransfer/message/types/message_types.go @@ -0,0 +1,16 @@ +package types + +type MessageType uint64 + +// Always append at the end to avoid breaking backward compatibility for cbor messages +const ( + NewMessage MessageType = iota + UpdateMessage + CancelMessage + CompleteMessage + VoucherMessage + VoucherResultMessage + + RestartMessage + RestartExistingChannelRequestMessage +) diff --git a/datatransfer/network/interface.go b/datatransfer/network/interface.go new file mode 100644 index 000000000..39d9af908 --- /dev/null +++ b/datatransfer/network/interface.go @@ -0,0 +1,57 @@ +package network + +import ( + "context" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/libp2p/go-libp2p/core/peer" + "github.com/libp2p/go-libp2p/core/protocol" +) + +// DataTransferNetwork provides network connectivity for GraphSync. +type DataTransferNetwork interface { + Protect(id peer.ID, tag string) + Unprotect(id peer.ID, tag string) bool + + // SendMessage sends a GraphSync message to a peer. + SendMessage( + context.Context, + peer.ID, + datatransfer.Message) error + + // SetDelegate registers the Reciver to handle messages received from the + // network. + SetDelegate(Receiver) + + // ConnectTo establishes a connection to the given peer + ConnectTo(context.Context, peer.ID) error + + // ConnectWithRetry establishes a connection to the given peer, retrying if + // necessary, and opens a stream on the data-transfer protocol to verify + // the peer will accept messages on the protocol + ConnectWithRetry(ctx context.Context, p peer.ID) error + + // ID returns the peer id of this libp2p host + ID() peer.ID + + // Protocol returns the protocol version of the peer, connecting to + // the peer if necessary + Protocol(context.Context, peer.ID) (protocol.ID, error) +} + +// Receiver is an interface for receiving messages from the GraphSyncNetwork. +type Receiver interface { + ReceiveRequest( + ctx context.Context, + sender peer.ID, + incoming datatransfer.Request) + + ReceiveResponse( + ctx context.Context, + sender peer.ID, + incoming datatransfer.Response) + + ReceiveRestartExistingChannelRequest(ctx context.Context, sender peer.ID, incoming datatransfer.Request) + + ReceiveError(error) +} diff --git a/datatransfer/network/libp2p_impl.go b/datatransfer/network/libp2p_impl.go new file mode 100644 index 000000000..1db400993 --- /dev/null +++ b/datatransfer/network/libp2p_impl.go @@ -0,0 +1,355 @@ +package network + +import ( + "context" + "fmt" + "io" + "time" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/message" + logging "github.com/ipfs/go-log/v2" + "github.com/jpillora/backoff" + "github.com/libp2p/go-libp2p/core/host" + "github.com/libp2p/go-libp2p/core/network" + "github.com/libp2p/go-libp2p/core/peer" + "github.com/libp2p/go-libp2p/core/protocol" + "go.opentelemetry.io/otel" + "go.opentelemetry.io/otel/attribute" + "go.opentelemetry.io/otel/codes" + "go.opentelemetry.io/otel/trace" + "golang.org/x/xerrors" +) + +var log = logging.Logger("data_transfer_network") + +// The maximum amount of time to wait to open a stream +const defaultOpenStreamTimeout = 10 * time.Second + +// The maximum time to wait for a message to be sent +var defaultSendMessageTimeout = 10 * time.Second + +// The max number of attempts to open a stream +const defaultMaxStreamOpenAttempts = 5 + +// The min backoff time between retries +const defaultMinAttemptDuration = 1 * time.Second + +// The max backoff time between retries +const defaultMaxAttemptDuration = 5 * time.Minute + +// The multiplier in the backoff time for each retry +const defaultBackoffFactor = 5 + +var defaultDataTransferProtocols = []protocol.ID{ + datatransfer.ProtocolDataTransfer1_2, +} + +// Option is an option for configuring the libp2p storage market network +type Option func(*libp2pDataTransferNetwork) + +// DataTransferProtocols OVERWRITES the default libp2p protocols we use for data transfer with the given protocols. +func DataTransferProtocols(protocols []protocol.ID) Option { + return func(impl *libp2pDataTransferNetwork) { + impl.setDataTransferProtocols(protocols) + } +} + +// SendMessageParameters changes the default parameters around sending messages +func SendMessageParameters(openStreamTimeout time.Duration, sendMessageTimeout time.Duration) Option { + return func(impl *libp2pDataTransferNetwork) { + impl.sendMessageTimeout = sendMessageTimeout + impl.openStreamTimeout = openStreamTimeout + } +} + +// RetryParameters changes the default parameters around connection reopening +func RetryParameters(minDuration time.Duration, maxDuration time.Duration, attempts float64, backoffFactor float64) Option { + return func(impl *libp2pDataTransferNetwork) { + impl.maxStreamOpenAttempts = attempts + impl.minAttemptDuration = minDuration + impl.maxAttemptDuration = maxDuration + impl.backoffFactor = backoffFactor + } +} + +// NewFromLibp2pHost returns a GraphSyncNetwork supported by underlying Libp2p host. +func NewFromLibp2pHost(host host.Host, options ...Option) DataTransferNetwork { + dataTransferNetwork := libp2pDataTransferNetwork{ + host: host, + + openStreamTimeout: defaultOpenStreamTimeout, + sendMessageTimeout: defaultSendMessageTimeout, + maxStreamOpenAttempts: defaultMaxStreamOpenAttempts, + minAttemptDuration: defaultMinAttemptDuration, + maxAttemptDuration: defaultMaxAttemptDuration, + backoffFactor: defaultBackoffFactor, + } + dataTransferNetwork.setDataTransferProtocols(defaultDataTransferProtocols) + + for _, option := range options { + option(&dataTransferNetwork) + } + + return &dataTransferNetwork +} + +// libp2pDataTransferNetwork transforms the libp2p host interface, which sends and receives +// NetMessage objects, into the data transfer network interface. +type libp2pDataTransferNetwork struct { + host host.Host + // inbound messages from the network are forwarded to the receiver + receiver Receiver + + openStreamTimeout time.Duration + sendMessageTimeout time.Duration + maxStreamOpenAttempts float64 + minAttemptDuration time.Duration + maxAttemptDuration time.Duration + dtProtocols []protocol.ID + dtProtocolStrings []string + backoffFactor float64 +} + +func (impl *libp2pDataTransferNetwork) openStream(ctx context.Context, id peer.ID, protocols ...protocol.ID) (network.Stream, error) { + b := &backoff.Backoff{ + Min: impl.minAttemptDuration, + Max: impl.maxAttemptDuration, + Factor: impl.backoffFactor, + Jitter: true, + } + + start := time.Now() + for { + tctx, cancel := context.WithTimeout(ctx, impl.openStreamTimeout) + defer cancel() + + // will use the first among the given protocols that the remote peer supports + at := time.Now() + s, err := impl.host.NewStream(tctx, id, protocols...) + if err == nil { + nAttempts := b.Attempt() + 1 + if b.Attempt() > 0 { + log.Debugf("opened stream to %s on attempt %g of %g after %s", + id, nAttempts, impl.maxStreamOpenAttempts, time.Since(start)) + } + + return s, err + } + + // b.Attempt() starts from zero + nAttempts := b.Attempt() + 1 + if nAttempts >= impl.maxStreamOpenAttempts { + return nil, xerrors.Errorf("exhausted %g attempts but failed to open stream to %s, err: %w", impl.maxStreamOpenAttempts, id, err) + } + + d := b.Duration() + log.Warnf("failed to open stream to %s on attempt %g of %g after %s, waiting %s to try again, err: %s", + id, nAttempts, impl.maxStreamOpenAttempts, time.Since(at), d, err) + + select { + case <-ctx.Done(): + return nil, ctx.Err() + case <-time.After(d): + } + } +} + +func (dtnet *libp2pDataTransferNetwork) SendMessage( + ctx context.Context, + p peer.ID, + outgoing datatransfer.Message) error { + + ctx, span := otel.Tracer("data-transfer").Start(ctx, "sendMessage", trace.WithAttributes( + attribute.String("to", p.String()), + attribute.Int64("transferID", int64(outgoing.TransferID())), + attribute.Bool("isRequest", outgoing.IsRequest()), + attribute.Bool("isNew", outgoing.IsNew()), + attribute.Bool("isRestart", outgoing.IsRestart()), + attribute.Bool("isUpdate", outgoing.IsUpdate()), + attribute.Bool("isCancel", outgoing.IsCancel()), + attribute.Bool("isPaused", outgoing.IsPaused()), + )) + + defer span.End() + s, err := dtnet.openStream(ctx, p, dtnet.dtProtocols...) + if err != nil { + span.RecordError(err) + span.SetStatus(codes.Error, err.Error()) + return err + } + + outgoing, err = outgoing.MessageForProtocol(s.Protocol()) + if err != nil { + err = xerrors.Errorf("failed to convert message for protocol: %w", err) + span.RecordError(err) + span.SetStatus(codes.Error, err.Error()) + return err + } + + if err = dtnet.msgToStream(ctx, s, outgoing); err != nil { + if err2 := s.Reset(); err2 != nil { + log.Error(err) + span.RecordError(err2) + span.SetStatus(codes.Error, err2.Error()) + return err2 + } + span.RecordError(err) + span.SetStatus(codes.Error, err.Error()) + return err + } + + return s.Close() +} + +func (dtnet *libp2pDataTransferNetwork) SetDelegate(r Receiver) { + dtnet.receiver = r + for _, p := range dtnet.dtProtocols { + dtnet.host.SetStreamHandler(p, dtnet.handleNewStream) + } +} + +func (dtnet *libp2pDataTransferNetwork) ConnectTo(ctx context.Context, p peer.ID) error { + return dtnet.host.Connect(ctx, peer.AddrInfo{ID: p}) +} + +// ConnectWithRetry establishes a connection to the given peer, retrying if +// necessary, and opens a stream on the data-transfer protocol to verify +// the peer will accept messages on the protocol +func (dtnet *libp2pDataTransferNetwork) ConnectWithRetry(ctx context.Context, p peer.ID) error { + // Open a stream over the data-transfer protocol, to make sure that the + // peer is listening on the protocol + s, err := dtnet.openStream(ctx, p, dtnet.dtProtocols...) + if err != nil { + return err + } + + // We don't actually use the stream, we just open it to verify it's + // possible to connect over the data-transfer protocol, so we close it here + return s.Close() +} + +// handleNewStream receives a new stream from the network. +func (dtnet *libp2pDataTransferNetwork) handleNewStream(s network.Stream) { + defer s.Close() // nolint: errcheck,gosec + + if dtnet.receiver == nil { + s.Reset() // nolint: errcheck,gosec + return + } + + p := s.Conn().RemotePeer() + for { + var received datatransfer.Message + var err error + switch s.Protocol() { + case datatransfer.ProtocolDataTransfer1_2: + received, err = message.FromNet(s) + } + + if err != nil { + if err != io.EOF && err != io.ErrUnexpectedEOF { + s.Reset() // nolint: errcheck,gosec + go dtnet.receiver.ReceiveError(err) + log.Debugf("net handleNewStream from %s error: %s", p, err) + } + return + } + + ctx := context.Background() + log.Debugf("net handleNewStream from %s", p) + + if received.IsRequest() { + receivedRequest, ok := received.(datatransfer.Request) + if ok { + if receivedRequest.IsRestartExistingChannelRequest() { + dtnet.receiver.ReceiveRestartExistingChannelRequest(ctx, p, receivedRequest) + } else { + dtnet.receiver.ReceiveRequest(ctx, p, receivedRequest) + } + } + } else { + receivedResponse, ok := received.(datatransfer.Response) + if ok { + dtnet.receiver.ReceiveResponse(ctx, p, receivedResponse) + } + } + } +} + +func (dtnet *libp2pDataTransferNetwork) ID() peer.ID { + return dtnet.host.ID() +} + +func (dtnet *libp2pDataTransferNetwork) Protect(id peer.ID, tag string) { + dtnet.host.ConnManager().Protect(id, tag) +} + +func (dtnet *libp2pDataTransferNetwork) Unprotect(id peer.ID, tag string) bool { + return dtnet.host.ConnManager().Unprotect(id, tag) +} + +func (dtnet *libp2pDataTransferNetwork) msgToStream(ctx context.Context, s network.Stream, msg datatransfer.Message) error { + if msg.IsRequest() { + log.Debugf("Outgoing request message for transfer ID: %d", msg.TransferID()) + } + + deadline := time.Now().Add(dtnet.sendMessageTimeout) + if dl, ok := ctx.Deadline(); ok { + deadline = dl + } + if err := s.SetWriteDeadline(deadline); err != nil { + log.Warnf("error setting deadline: %s", err) + } + defer func() { + if err := s.SetWriteDeadline(time.Time{}); err != nil { + log.Warnf("error resetting deadline: %s", err) + } + }() + + switch s.Protocol() { + case datatransfer.ProtocolDataTransfer1_2: + default: + return fmt.Errorf("unrecognized protocol on remote: %s", s.Protocol()) + } + + if err := msg.ToNet(s); err != nil { + log.Debugf("error: %s", err) + return err + } + + return nil +} + +func (impl *libp2pDataTransferNetwork) Protocol(ctx context.Context, id peer.ID) (protocol.ID, error) { + // Check the cache for the peer's protocol version + firstProto, err := impl.host.Peerstore().FirstSupportedProtocol(id, impl.dtProtocols...) + if err != nil { + return "", err + } + + if firstProto != "" { + return protocol.ID(firstProto), nil + } + + // The peer's protocol version is not in the cache, so connect to the peer. + // Note that when the stream is opened, the peer's protocol will be added + // to the cache. + s, err := impl.openStream(ctx, id, impl.dtProtocols...) + if err != nil { + return "", err + } + _ = s.Close() + + return s.Protocol(), nil +} + +func (impl *libp2pDataTransferNetwork) setDataTransferProtocols(protocols []protocol.ID) { + impl.dtProtocols = append([]protocol.ID{}, protocols...) + + // Keep a string version of the protocols for performance reasons + impl.dtProtocolStrings = make([]string, 0, len(impl.dtProtocols)) + for _, proto := range impl.dtProtocols { + impl.dtProtocolStrings = append(impl.dtProtocolStrings, string(proto)) + } +} diff --git a/datatransfer/registry/registry.go b/datatransfer/registry/registry.go new file mode 100644 index 000000000..2cdc7ff0d --- /dev/null +++ b/datatransfer/registry/registry.go @@ -0,0 +1,81 @@ +package registry + +import ( + "sync" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/encoding" + "golang.org/x/xerrors" +) + +// Processor is an interface that processes a certain type of encodable objects +// in a registry. The actual specifics of the interface that must be satisfied are +// left to the user of the registry +type Processor interface{} + +type registryEntry struct { + decoder encoding.Decoder + processor Processor +} + +// Registry maintans a register of types of encodable objects and a corresponding +// processor for those objects +// The encodable types must have a method Type() that specifies and identifier +// so they correct decoding function and processor can be identified based +// on this unique identifier +type Registry struct { + registryLk sync.RWMutex + entries map[datatransfer.TypeIdentifier]registryEntry +} + +// NewRegistry initialzes a new registy +func NewRegistry() *Registry { + return &Registry{ + entries: make(map[datatransfer.TypeIdentifier]registryEntry), + } +} + +// Register registers the given processor for the given entry type +func (r *Registry) Register(entry datatransfer.Registerable, processor Processor) error { + identifier := entry.Type() + decoder, err := encoding.NewDecoder(entry) + if err != nil { + return xerrors.Errorf("registering entry type %s: %w", identifier, err) + } + r.registryLk.Lock() + defer r.registryLk.Unlock() + if _, ok := r.entries[identifier]; ok { + return xerrors.Errorf("identifier already registered: %s", identifier) + } + r.entries[identifier] = registryEntry{decoder, processor} + return nil +} + +// Decoder gets a decoder for the given identifier +func (r *Registry) Decoder(identifier datatransfer.TypeIdentifier) (encoding.Decoder, bool) { + r.registryLk.RLock() + entry, has := r.entries[identifier] + r.registryLk.RUnlock() + return entry.decoder, has +} + +// Processor gets the processing interface for the given identifer +func (r *Registry) Processor(identifier datatransfer.TypeIdentifier) (Processor, bool) { + r.registryLk.RLock() + entry, has := r.entries[identifier] + r.registryLk.RUnlock() + return entry.processor, has +} + +// Each iterates through all of the entries in this registry +func (r *Registry) Each(process func(datatransfer.TypeIdentifier, encoding.Decoder, Processor) error) error { + r.registryLk.RLock() + defer r.registryLk.RUnlock() + for identifier, entry := range r.entries { + err := process(identifier, entry.decoder, entry.processor) + if err != nil { + return err + } + } + return nil +} diff --git a/datatransfer/testutil/testutils.go b/datatransfer/testutil/testutils.go new file mode 100644 index 000000000..d4606fc24 --- /dev/null +++ b/datatransfer/testutil/testutils.go @@ -0,0 +1,24 @@ +package testutil + +import ( + "testing" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/stretchr/testify/require" + "golang.org/x/net/context" +) + +// StartAndWaitForReady is a utility function to start a module and verify it reaches the ready state +func StartAndWaitForReady(ctx context.Context, t *testing.T, manager datatransfer.Manager) { + ready := make(chan error, 1) + manager.OnReady(func(err error) { + ready <- err + }) + require.NoError(t, manager.Start(ctx)) + select { + case <-ctx.Done(): + t.Fatal("did not finish starting up module") + case err := <-ready: + require.NoError(t, err) + } +} diff --git a/datatransfer/tracing/tracing.go b/datatransfer/tracing/tracing.go new file mode 100644 index 000000000..d43be30b7 --- /dev/null +++ b/datatransfer/tracing/tracing.go @@ -0,0 +1,64 @@ +package tracing + +import ( + "context" + "sync" + + "go.opentelemetry.io/otel" + "go.opentelemetry.io/otel/attribute" + "go.opentelemetry.io/otel/trace" + + "github.com/filecoin-project/boost/datatransfer" +) + +type SpansIndex struct { + spansLk sync.RWMutex + spans map[datatransfer.ChannelID]trace.Span +} + +func NewSpansIndex() *SpansIndex { + return &SpansIndex{ + spans: make(map[datatransfer.ChannelID]trace.Span), + } +} + +func (si *SpansIndex) SpanForChannel(ctx context.Context, chid datatransfer.ChannelID) (context.Context, trace.Span) { + si.spansLk.RLock() + span, ok := si.spans[chid] + si.spansLk.RUnlock() + if ok { + return trace.ContextWithSpan(ctx, span), span + } + si.spansLk.Lock() + defer si.spansLk.Unlock() + // need to recheck under the write lock + span, ok = si.spans[chid] + if ok { + return trace.ContextWithSpan(ctx, span), span + } + ctx, span = otel.Tracer("data-transfer").Start(ctx, "transfer", trace.WithAttributes( + attribute.String("channelID", chid.String()), + )) + si.spans[chid] = span + return ctx, span +} + +func (si *SpansIndex) EndChannelSpan(chid datatransfer.ChannelID) { + si.spansLk.Lock() + defer si.spansLk.Unlock() + span, ok := si.spans[chid] + if ok { + span.End() + delete(si.spans, chid) + } +} + +func (si *SpansIndex) EndAll() { + si.spansLk.Lock() + defer si.spansLk.Unlock() + for _, span := range si.spans { + span.End() + } + // reset in case someone continues to use the span index + si.spans = make(map[datatransfer.ChannelID]trace.Span) +} diff --git a/datatransfer/transport/graphsync/extension/gsextension.go b/datatransfer/transport/graphsync/extension/gsextension.go new file mode 100644 index 000000000..3c49cf86a --- /dev/null +++ b/datatransfer/transport/graphsync/extension/gsextension.go @@ -0,0 +1,83 @@ +package extension + +import ( + "errors" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/message" + "github.com/ipld/go-ipld-prime/datamodel" + "github.com/libp2p/go-libp2p/core/protocol" + + graphsync "github.com/filecoin-project/boost-graphsync" +) + +const ( + // ExtensionIncomingRequest1_1 is the identifier for data sent by the IncomingRequest hook + ExtensionIncomingRequest1_1 = graphsync.ExtensionName("fil/data-transfer/incoming-request/1.1") + // ExtensionOutgoingBlock1_1 is the identifier for data sent by the OutgoingBlock hook + ExtensionOutgoingBlock1_1 = graphsync.ExtensionName("fil/data-transfer/outgoing-block/1.1") + // ExtensionDataTransfer1_1 is the identifier for the v1.1 data transfer extension to graphsync + ExtensionDataTransfer1_1 = graphsync.ExtensionName("fil/data-transfer/1.1") +) + +// ProtocolMap maps graphsync extensions to their libp2p protocols +var ProtocolMap = map[graphsync.ExtensionName]protocol.ID{ + ExtensionIncomingRequest1_1: datatransfer.ProtocolDataTransfer1_2, + ExtensionOutgoingBlock1_1: datatransfer.ProtocolDataTransfer1_2, + ExtensionDataTransfer1_1: datatransfer.ProtocolDataTransfer1_2, +} + +// ToExtensionData converts a message to a graphsync extension +func ToExtensionData(msg datatransfer.Message, supportedExtensions []graphsync.ExtensionName) ([]graphsync.ExtensionData, error) { + exts := make([]graphsync.ExtensionData, 0, len(supportedExtensions)) + for _, supportedExtension := range supportedExtensions { + protoID, ok := ProtocolMap[supportedExtension] + if !ok { + return nil, errors.New("unsupported protocol") + } + versionedMsg, err := msg.MessageForProtocol(protoID) + if err != nil { + continue + } + nd, err := versionedMsg.ToIPLD() + if err != nil { + return nil, err + } + exts = append(exts, graphsync.ExtensionData{ + Name: supportedExtension, + Data: nd, + }) + } + if len(exts) == 0 { + return nil, errors.New("message not encodable in any supported extensions") + } + return exts, nil +} + +// GsExtended is a small interface used by GetTransferData +type GsExtended interface { + Extension(name graphsync.ExtensionName) (datamodel.Node, bool) +} + +// GetTransferData unmarshals extension data. +// Returns: +// - nil + nil if the extension is not found +// - nil + error if the extendedData fails to unmarshal +// - unmarshaled ExtensionDataTransferData + nil if all goes well +func GetTransferData(extendedData GsExtended, extNames []graphsync.ExtensionName) (datatransfer.Message, error) { + for _, name := range extNames { + data, ok := extendedData.Extension(name) + if ok { + return decoders[name](data) + } + } + return nil, nil +} + +type decoder func(datamodel.Node) (datatransfer.Message, error) + +var decoders = map[graphsync.ExtensionName]decoder{ + ExtensionIncomingRequest1_1: message.FromIPLD, + ExtensionOutgoingBlock1_1: message.FromIPLD, + ExtensionDataTransfer1_1: message.FromIPLD, +} diff --git a/datatransfer/transport/graphsync/graphsync.go b/datatransfer/transport/graphsync/graphsync.go new file mode 100644 index 000000000..b3f8e0058 --- /dev/null +++ b/datatransfer/transport/graphsync/graphsync.go @@ -0,0 +1,1300 @@ +package graphsync + +import ( + "context" + "errors" + "fmt" + "sync" + "time" + + graphsync "github.com/filecoin-project/boost-graphsync" + "github.com/filecoin-project/boost-graphsync/donotsendfirstblocks" + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/transport/graphsync/extension" + logging "github.com/ipfs/go-log/v2" + "github.com/ipld/go-ipld-prime" + "github.com/libp2p/go-libp2p/core/peer" + "golang.org/x/sync/errgroup" +) + +var log = logging.Logger("dt_graphsync") + +// When restarting a data transfer, we cancel the existing graphsync request +// before opening a new one. +// This constant defines the maximum time to wait for the request to be +// cancelled. +const maxGSCancelWait = time.Second + +var defaultSupportedExtensions = []graphsync.ExtensionName{ + extension.ExtensionDataTransfer1_1, +} + +var incomingReqExtensions = []graphsync.ExtensionName{ + extension.ExtensionIncomingRequest1_1, + extension.ExtensionDataTransfer1_1, +} + +var outgoingBlkExtensions = []graphsync.ExtensionName{ + extension.ExtensionOutgoingBlock1_1, + extension.ExtensionDataTransfer1_1, +} + +// Option is an option for setting up the graphsync transport +type Option func(*Transport) + +// SupportedExtensions sets what data transfer extensions are supported +func SupportedExtensions(supportedExtensions []graphsync.ExtensionName) Option { + return func(t *Transport) { + t.supportedExtensions = supportedExtensions + } +} + +// RegisterCompletedRequestListener is used by the tests +func RegisterCompletedRequestListener(l func(channelID datatransfer.ChannelID)) Option { + return func(t *Transport) { + t.completedRequestListener = l + } +} + +// RegisterCompletedResponseListener is used by the tests +func RegisterCompletedResponseListener(l func(channelID datatransfer.ChannelID)) Option { + return func(t *Transport) { + t.completedResponseListener = l + } +} + +// Transport manages graphsync hooks for data transfer, translating from +// graphsync hooks to semantic data transfer events +type Transport struct { + events datatransfer.EventsHandler + gs graphsync.GraphExchange + peerID peer.ID + + supportedExtensions []graphsync.ExtensionName + unregisterFuncs []graphsync.UnregisterHookFunc + completedRequestListener func(channelID datatransfer.ChannelID) + completedResponseListener func(channelID datatransfer.ChannelID) + + // Map from data transfer channel ID to information about that channel + dtChannelsLk sync.RWMutex + dtChannels map[datatransfer.ChannelID]*dtChannel + + // Used in graphsync callbacks to map from graphsync request to the + // associated data-transfer channel ID. + requestIDToChannelID *requestIDToChannelIDMap +} + +// NewTransport makes a new hooks manager with the given hook events interface +func NewTransport(peerID peer.ID, gs graphsync.GraphExchange, options ...Option) *Transport { + t := &Transport{ + gs: gs, + peerID: peerID, + supportedExtensions: defaultSupportedExtensions, + dtChannels: make(map[datatransfer.ChannelID]*dtChannel), + requestIDToChannelID: newRequestIDToChannelIDMap(), + } + for _, option := range options { + option(t) + } + return t +} + +// OpenChannel initiates an outgoing request for the other peer to send data +// to us on this channel +// Note: from a data transfer symantic standpoint, it doesn't matter if the +// request is push or pull -- OpenChannel is called by the party that is +// intending to receive data +func (t *Transport) OpenChannel( + ctx context.Context, + dataSender peer.ID, + channelID datatransfer.ChannelID, + root ipld.Link, + stor ipld.Node, + channel datatransfer.ChannelState, + msg datatransfer.Message, +) error { + if t.events == nil { + return datatransfer.ErrHandlerNotSet + } + + exts, err := extension.ToExtensionData(msg, t.supportedExtensions) + if err != nil { + return err + } + // If this is a restart request, the client can indicate the blocks that + // it has already received, so that the provider knows not to resend + // those blocks + restartExts, err := t.getRestartExtension(ctx, dataSender, channel) + if err != nil { + return err + } + exts = append(exts, restartExts...) + + // Start tracking the data-transfer channel + ch := t.trackDTChannel(channelID) + + // Open a graphsync request to the remote peer + req, err := ch.open(ctx, channelID, dataSender, root, stor, channel, exts) + if err != nil { + return err + } + + // Process incoming data + go t.executeGsRequest(req) + + return nil +} + +// Get the extension data for sending a Restart message, depending on the +// protocol version of the peer +func (t *Transport) getRestartExtension(ctx context.Context, p peer.ID, channel datatransfer.ChannelState) ([]graphsync.ExtensionData, error) { + if channel == nil { + return nil, nil + } + return getDoNotSendFirstBlocksExtension(channel) +} + +// Skip the first N blocks because they were already received +func getDoNotSendFirstBlocksExtension(channel datatransfer.ChannelState) ([]graphsync.ExtensionData, error) { + skipBlockCount := channel.ReceivedCidsTotal() + data := donotsendfirstblocks.EncodeDoNotSendFirstBlocks(skipBlockCount) + return []graphsync.ExtensionData{{ + Name: graphsync.ExtensionsDoNotSendFirstBlocks, + Data: data, + }}, nil +} + +// Read from the graphsync response and error channels until they are closed, +// and return the last error on the error channel +func (t *Transport) consumeResponses(req *gsReq) error { + var lastError error + for range req.responseChan { + } + log.Debugf("channel %s: finished consuming graphsync response channel", req.channelID) + + for err := range req.errChan { + lastError = err + } + log.Debugf("channel %s: finished consuming graphsync error channel", req.channelID) + + return lastError +} + +// Read from the graphsync response and error channels until they are closed +// or there is an error, then call the channel completed callback +func (t *Transport) executeGsRequest(req *gsReq) { + // Make sure to call the onComplete callback before returning + defer func() { + log.Infow("gs request complete for channel", "chid", req.channelID) + req.onComplete() + }() + + // Consume the response and error channels for the graphsync request + lastError := t.consumeResponses(req) + + // Request cancelled by client + if _, ok := lastError.(graphsync.RequestClientCancelledErr); ok { + terr := fmt.Errorf("graphsync request cancelled") + log.Warnf("channel %s: %s", req.channelID, terr) + if err := t.events.OnRequestCancelled(req.channelID, terr); err != nil { + log.Error(err) + } + return + } + + // Request cancelled by responder + if _, ok := lastError.(graphsync.RequestCancelledErr); ok { + log.Infof("channel %s: graphsync request cancelled by responder", req.channelID) + // TODO Should we do anything for RequestCancelledErr ? + return + } + + if lastError != nil { + log.Warnf("channel %s: graphsync error: %s", req.channelID, lastError) + } + + log.Debugf("channel %s: finished executing graphsync request", req.channelID) + + var completeErr error + if lastError != nil { + completeErr = fmt.Errorf("channel %s: graphsync request failed to complete: %w", req.channelID, lastError) + } + + // Used by the tests to listen for when a request completes + if t.completedRequestListener != nil { + t.completedRequestListener(req.channelID) + } + + err := t.events.OnChannelCompleted(req.channelID, completeErr) + if err != nil { + log.Errorf("channel %s: processing OnChannelCompleted: %s", req.channelID, err) + } +} + +// PauseChannel pauses the given data-transfer channel +func (t *Transport) PauseChannel(ctx context.Context, chid datatransfer.ChannelID) error { + ch, err := t.getDTChannel(chid) + if err != nil { + return err + } + return ch.pause(ctx) +} + +// ResumeChannel resumes the given data-transfer channel and sends the message +// if there is one +func (t *Transport) ResumeChannel( + ctx context.Context, + msg datatransfer.Message, + chid datatransfer.ChannelID, +) error { + ch, err := t.getDTChannel(chid) + if err != nil { + return err + } + return ch.resume(ctx, msg) +} + +// CloseChannel closes the given data-transfer channel +func (t *Transport) CloseChannel(ctx context.Context, chid datatransfer.ChannelID) error { + ch, err := t.getDTChannel(chid) + if err != nil { + return err + } + + err = ch.close(ctx) + if err != nil { + return fmt.Errorf("closing channel: %w", err) + } + return nil +} + +// CleanupChannel is called on the otherside of a cancel - removes any associated +// data for the channel +func (t *Transport) CleanupChannel(chid datatransfer.ChannelID) { + t.dtChannelsLk.Lock() + + ch, ok := t.dtChannels[chid] + if ok { + // Remove the reference to the channel from the channels map + delete(t.dtChannels, chid) + } + + t.dtChannelsLk.Unlock() + + // Clean up the channel + if ok { + ch.cleanup() + } +} + +// SetEventHandler sets the handler for events on channels +func (t *Transport) SetEventHandler(events datatransfer.EventsHandler) error { + if t.events != nil { + return datatransfer.ErrHandlerAlreadySet + } + t.events = events + + t.unregisterFuncs = append(t.unregisterFuncs, t.gs.RegisterIncomingRequestQueuedHook(t.gsReqQueuedHook)) + t.unregisterFuncs = append(t.unregisterFuncs, t.gs.RegisterIncomingRequestHook(t.gsReqRecdHook)) + t.unregisterFuncs = append(t.unregisterFuncs, t.gs.RegisterCompletedResponseListener(t.gsCompletedResponseListener)) + t.unregisterFuncs = append(t.unregisterFuncs, t.gs.RegisterIncomingBlockHook(t.gsIncomingBlockHook)) + t.unregisterFuncs = append(t.unregisterFuncs, t.gs.RegisterOutgoingBlockHook(t.gsOutgoingBlockHook)) + t.unregisterFuncs = append(t.unregisterFuncs, t.gs.RegisterBlockSentListener(t.gsBlockSentHook)) + t.unregisterFuncs = append(t.unregisterFuncs, t.gs.RegisterOutgoingRequestHook(t.gsOutgoingRequestHook)) + t.unregisterFuncs = append(t.unregisterFuncs, t.gs.RegisterIncomingResponseHook(t.gsIncomingResponseHook)) + t.unregisterFuncs = append(t.unregisterFuncs, t.gs.RegisterRequestUpdatedHook(t.gsRequestUpdatedHook)) + t.unregisterFuncs = append(t.unregisterFuncs, t.gs.RegisterRequestorCancelledListener(t.gsRequestorCancelledListener)) + t.unregisterFuncs = append(t.unregisterFuncs, t.gs.RegisterNetworkErrorListener(t.gsNetworkSendErrorListener)) + t.unregisterFuncs = append(t.unregisterFuncs, t.gs.RegisterReceiverNetworkErrorListener(t.gsNetworkReceiveErrorListener)) + return nil +} + +// Shutdown disconnects a transport interface from graphsync +func (t *Transport) Shutdown(ctx context.Context) error { + for _, unregisterFunc := range t.unregisterFuncs { + unregisterFunc() + } + + t.dtChannelsLk.Lock() + defer t.dtChannelsLk.Unlock() + + var eg errgroup.Group + for _, ch := range t.dtChannels { + ch := ch + eg.Go(func() error { + return ch.shutdown(ctx) + }) + } + + err := eg.Wait() + if err != nil { + return fmt.Errorf("shutting down graphsync transport: %w", err) + } + return nil +} + +// UseStore tells the graphsync transport to use the given loader and storer for this channelID +func (t *Transport) UseStore(channelID datatransfer.ChannelID, lsys ipld.LinkSystem) error { + ch := t.trackDTChannel(channelID) + return ch.useStore(lsys) +} + +// ChannelGraphsyncRequests describes any graphsync request IDs associated with a given channel +type ChannelGraphsyncRequests struct { + // Current is the current request ID for the transfer + Current graphsync.RequestID + // Previous are ids of previous GraphSync requests in a transfer that + // has been restarted. We may be interested to know if these IDs are active + // on either side of the request + Previous []graphsync.RequestID +} + +// ChannelsForPeer describes current active channels for a given peer and their +// associated graphsync requests +type ChannelsForPeer struct { + SendingChannels map[datatransfer.ChannelID]ChannelGraphsyncRequests + ReceivingChannels map[datatransfer.ChannelID]ChannelGraphsyncRequests +} + +// ChannelsForPeer identifies which channels are open and which request IDs they map to +func (t *Transport) ChannelsForPeer(p peer.ID) ChannelsForPeer { + t.dtChannelsLk.RLock() + defer t.dtChannelsLk.RUnlock() + + // cannot have active transfers with self + if p == t.peerID { + return ChannelsForPeer{ + SendingChannels: map[datatransfer.ChannelID]ChannelGraphsyncRequests{}, + ReceivingChannels: map[datatransfer.ChannelID]ChannelGraphsyncRequests{}, + } + } + + sending := make(map[datatransfer.ChannelID]ChannelGraphsyncRequests) + receiving := make(map[datatransfer.ChannelID]ChannelGraphsyncRequests) + // loop through every graphsync request key we're currently tracking + t.requestIDToChannelID.forEach(func(requestID graphsync.RequestID, isSending bool, chid datatransfer.ChannelID) { + // if the associated channel ID includes the requested peer + if chid.Initiator == p || chid.Responder == p { + // determine whether the requested peer is one at least one end of the channel + // and whether we're receving from that peer or sending to it + collection := sending + if !isSending { + collection = receiving + } + channelGraphsyncRequests := collection[chid] + // finally, determine if the request key matches the current GraphSync key we're tracking for + // this channel, indicating it's the current graphsync request + if t.dtChannels[chid] != nil && t.dtChannels[chid].requestID != nil && (*t.dtChannels[chid].requestID) == requestID { + channelGraphsyncRequests.Current = requestID + } else { + // otherwise this id was a previous graphsync request on a channel that was restarted + // and it has not been cleaned up yet + channelGraphsyncRequests.Previous = append(channelGraphsyncRequests.Previous, requestID) + } + collection[chid] = channelGraphsyncRequests + } + }) + return ChannelsForPeer{ + SendingChannels: sending, + ReceivingChannels: receiving, + } +} + +// gsOutgoingRequestHook is called when a graphsync request is made +func (t *Transport) gsOutgoingRequestHook(p peer.ID, request graphsync.RequestData, hookActions graphsync.OutgoingRequestHookActions) { + message, _ := extension.GetTransferData(request, t.supportedExtensions) + + // extension not found; probably not our request. + if message == nil { + return + } + + // A graphsync request is made when either + // - The local node opens a data-transfer pull channel, so the local node + // sends a graphsync request to ask the remote peer for the data + // - The remote peer opened a data-transfer push channel, and in response + // the local node sends a graphsync request to ask for the data + var initiator peer.ID + var responder peer.ID + if message.IsRequest() { + // This is a pull request so the data-transfer initiator is the local node + initiator = t.peerID + responder = p + } else { + // This is a push response so the data-transfer initiator is the remote + // peer: They opened the push channel, we respond by sending a + // graphsync request for the data + initiator = p + responder = t.peerID + } + chid := datatransfer.ChannelID{Initiator: initiator, Responder: responder, ID: message.TransferID()} + + // A data transfer channel was opened + err := t.events.OnChannelOpened(chid) + if err != nil { + // There was an error opening the channel, bail out + log.Errorf("processing OnChannelOpened for %s: %s", chid, err) + t.CleanupChannel(chid) + return + } + + // Start tracking the channel if we're not already + ch := t.trackDTChannel(chid) + + // Signal that the channel has been opened + ch.gsReqOpened(request.ID(), hookActions) +} + +// gsIncomingBlockHook is called when a block is received +func (t *Transport) gsIncomingBlockHook(p peer.ID, response graphsync.ResponseData, block graphsync.BlockData, hookActions graphsync.IncomingBlockHookActions) { + chid, ok := t.requestIDToChannelID.load(response.RequestID()) + if !ok { + return + } + + err := t.events.OnDataReceived(chid, block.Link(), block.BlockSize(), block.Index(), block.BlockSizeOnWire() != 0) + if err != nil && err != datatransfer.ErrPause { + hookActions.TerminateWithError(err) + return + } + + if err == datatransfer.ErrPause { + hookActions.PauseRequest() + } +} + +func (t *Transport) gsBlockSentHook(p peer.ID, request graphsync.RequestData, block graphsync.BlockData) { + // When a data transfer is restarted, the requester sends a list of CIDs + // that it already has. Graphsync calls the sent hook for all blocks even + // if they are in the list (meaning, they aren't actually sent over the + // wire). So here we check if the block was actually sent + // over the wire before firing the data sent event. + if block.BlockSizeOnWire() == 0 { + return + } + + chid, ok := t.requestIDToChannelID.load(request.ID()) + if !ok { + return + } + + if err := t.events.OnDataSent(chid, block.Link(), block.BlockSize(), block.Index(), block.BlockSizeOnWire() != 0); err != nil { + log.Errorf("failed to process data sent: %+v", err) + } +} + +func (t *Transport) gsOutgoingBlockHook(p peer.ID, request graphsync.RequestData, block graphsync.BlockData, hookActions graphsync.OutgoingBlockHookActions) { + // When a data transfer is restarted, the requester sends a list of CIDs + // that it already has. Graphsync calls the outgoing block hook for all + // blocks even if they are in the list (meaning, they aren't actually going + // to be sent over the wire). So here we check if the block is actually + // going to be sent over the wire before firing the data queued event. + if block.BlockSizeOnWire() == 0 { + return + } + + chid, ok := t.requestIDToChannelID.load(request.ID()) + if !ok { + return + } + + // OnDataQueued is called when a block is queued to be sent to the remote + // peer. It can return ErrPause to pause the response (eg if payment is + // required) and it can return a message that will be sent with the block + // (eg to ask for payment). + msg, err := t.events.OnDataQueued(chid, block.Link(), block.BlockSize(), block.Index(), block.BlockSizeOnWire() != 0) + if err != nil && err != datatransfer.ErrPause { + hookActions.TerminateWithError(err) + return + } + + if err == datatransfer.ErrPause { + hookActions.PauseResponse() + } + + if msg != nil { + // gsOutgoingBlockHook uses a unique extension name so it can be attached with data from a different hook + // outgoingBlkExtensions also includes the default extension name so it remains compatible with all data-transfer protocol versions out there + extensions, err := extension.ToExtensionData(msg, outgoingBlkExtensions) + if err != nil { + hookActions.TerminateWithError(err) + return + } + for _, extension := range extensions { + hookActions.SendExtensionData(extension) + } + } +} + +// gsReqQueuedHook is called when graphsync enqueues an incoming request for data +func (t *Transport) gsReqQueuedHook(p peer.ID, request graphsync.RequestData, hookActions graphsync.RequestQueuedHookActions) { + msg, err := extension.GetTransferData(request, t.supportedExtensions) + if err != nil { + log.Errorf("failed GetTransferData, req=%+v, err=%s", request, err) + } + // extension not found; probably not our request. + if msg == nil { + return + } + + var chid datatransfer.ChannelID + if msg.IsRequest() { + // when a data transfer request comes in on graphsync, the remote peer + // initiated a pull + chid = datatransfer.ChannelID{ID: msg.TransferID(), Initiator: p, Responder: t.peerID} + dtRequest := msg.(datatransfer.Request) + if dtRequest.IsNew() { + log.Infof("%s, pull request queued, req_id=%d", chid, request.ID()) + t.events.OnTransferQueued(chid) + } else { + log.Infof("%s, pull restart request queued, req_id=%d", chid, request.ID()) + } + } else { + // when a data transfer response comes in on graphsync, this node + // initiated a push, and the remote peer responded with a request + // for data + chid = datatransfer.ChannelID{ID: msg.TransferID(), Initiator: t.peerID, Responder: p} + response := msg.(datatransfer.Response) + if response.IsNew() { + log.Infof("%s, GS pull request queued in response to our push, req_id=%d", chid, request.ID()) + t.events.OnTransferQueued(chid) + } else { + log.Infof("%s, GS pull request queued in response to our restart push, req_id=%d", chid, request.ID()) + } + } + augmentContext := t.events.OnContextAugment(chid) + if augmentContext != nil { + hookActions.AugmentContext(augmentContext) + } +} + +// gsReqRecdHook is called when graphsync receives an incoming request for data +func (t *Transport) gsReqRecdHook(p peer.ID, request graphsync.RequestData, hookActions graphsync.IncomingRequestHookActions) { + log.Infow("received incoming request", "request", request) + + // if this is a push request the sender is us. + msg, err := extension.GetTransferData(request, t.supportedExtensions) + if err != nil { + log.Debugw("failed GetTransferData", "request", request, "err", err) + hookActions.TerminateWithError(err) + return + } + + // extension not found; probably not our request. + if msg == nil { + log.Debugw("no transfer data", "request", request) + return + } + + // An incoming graphsync request for data is received when either + // - The remote peer opened a data-transfer pull channel, so the local node + // receives a graphsync request for the data + // - The local node opened a data-transfer push channel, and in response + // the remote peer sent a graphsync request for the data, and now the + // local node receives that request for data + var chid datatransfer.ChannelID + var responseMessage datatransfer.Message + var ch *dtChannel + if msg.IsRequest() { + // when a data transfer request comes in on graphsync, the remote peer + // initiated a pull + chid = datatransfer.ChannelID{ID: msg.TransferID(), Initiator: p, Responder: t.peerID} + + log.Debugf("%s: received request for data (pull), req_id=%d", chid, request.ID()) + + // Lock the channel for the duration of this method + ch = t.trackDTChannel(chid) + ch.lk.Lock() + defer ch.lk.Unlock() + + request := msg.(datatransfer.Request) + responseMessage, err = t.events.OnRequestReceived(chid, request) + } else { + // when a data transfer response comes in on graphsync, this node + // initiated a push, and the remote peer responded with a request + // for data + chid = datatransfer.ChannelID{ID: msg.TransferID(), Initiator: t.peerID, Responder: p} + + log.Debugf("%s: received request for data (push), req_id=%d", chid, request.ID()) + + // Lock the channel for the duration of this method + ch = t.trackDTChannel(chid) + ch.lk.Lock() + defer ch.lk.Unlock() + + response := msg.(datatransfer.Response) + err = t.events.OnResponseReceived(chid, response) + } + + // If we need to send a response, add the response message as an extension + if responseMessage != nil { + // gsReqRecdHook uses a unique extension name so it can be attached with data from a different hook + // incomingReqExtensions also includes default extension name so it remains compatible with previous data-transfer + // protocol versions out there. + extensions, extensionErr := extension.ToExtensionData(responseMessage, incomingReqExtensions) + if extensionErr != nil { + log.Debugw("failed to convert extension data", "err", extensionErr) + hookActions.TerminateWithError(err) + return + } + for _, extension := range extensions { + hookActions.SendExtensionData(extension) + } + } + + if err != nil && err != datatransfer.ErrPause { + log.Debugw("failed to process request", "err", err) + hookActions.TerminateWithError(err) + return + } + + // Check if the callback indicated that the channel should be paused + // immediately (eg because data is still being unsealed) + paused := false + if err == datatransfer.ErrPause { + log.Debugf("%s: pausing graphsync response", chid) + + paused = true + hookActions.PauseResponse() + } + + // If this is a restart request, and the data transfer still hasn't got + // out of the paused state (eg because we're still unsealing), start this + // graphsync response in the paused state. + if ch.isOpen && !ch.xferStarted && !paused { + log.Debugf("%s: pausing graphsync response after restart", chid) + + paused = true + hookActions.PauseResponse() + } + + // If the transfer is not paused, record that the transfer has started + if !paused { + ch.xferStarted = true + } + + ch.gsDataRequestRcvd(request.ID(), hookActions) + + hookActions.ValidateRequest() +} + +// gsCompletedResponseListener is a graphsync.OnCompletedResponseListener. We use it learn when the data transfer is complete +// for the side that is responding to a graphsync request +func (t *Transport) gsCompletedResponseListener(p peer.ID, request graphsync.RequestData, status graphsync.ResponseStatusCode) { + chid, ok := t.requestIDToChannelID.load(request.ID()) + if !ok { + return + } + + if status == graphsync.RequestCancelled { + return + } + + var completeErr error + if status != graphsync.RequestCompletedFull { + statusStr := gsResponseStatusCodeString(status) + completeErr = fmt.Errorf("graphsync response to peer %s did not complete: response status code %s", p, statusStr) + } + + // Used by the tests to listen for when a response completes + if t.completedResponseListener != nil { + t.completedResponseListener(chid) + } + + err := t.events.OnChannelCompleted(chid, completeErr) + if err != nil { + log.Error(err) + } +} + +// Remove this map once this PR lands: https://github.com/ipfs/go-graphsync/pull/148 +var gsResponseStatusCodes = map[graphsync.ResponseStatusCode]string{ + graphsync.RequestAcknowledged: "RequestAcknowledged", + graphsync.AdditionalPeers: "AdditionalPeers", + graphsync.NotEnoughGas: "NotEnoughGas", + graphsync.OtherProtocol: "OtherProtocol", + graphsync.PartialResponse: "PartialResponse", + graphsync.RequestPaused: "RequestPaused", + graphsync.RequestCompletedFull: "RequestCompletedFull", + graphsync.RequestCompletedPartial: "RequestCompletedPartial", + graphsync.RequestRejected: "RequestRejected", + graphsync.RequestFailedBusy: "RequestFailedBusy", + graphsync.RequestFailedUnknown: "RequestFailedUnknown", + graphsync.RequestFailedLegal: "RequestFailedLegal", + graphsync.RequestFailedContentNotFound: "RequestFailedContentNotFound", + graphsync.RequestCancelled: "RequestCancelled", +} + +func gsResponseStatusCodeString(code graphsync.ResponseStatusCode) string { + str, ok := gsResponseStatusCodes[code] + if ok { + return str + } + return gsResponseStatusCodes[graphsync.RequestFailedUnknown] +} + +func (t *Transport) gsRequestUpdatedHook(p peer.ID, request graphsync.RequestData, update graphsync.RequestData, hookActions graphsync.RequestUpdatedHookActions) { + chid, ok := t.requestIDToChannelID.load(request.ID()) + if !ok { + return + } + + responseMessage, err := t.processExtension(chid, update, p, t.supportedExtensions) + + if responseMessage != nil { + extensions, extensionErr := extension.ToExtensionData(responseMessage, t.supportedExtensions) + if extensionErr != nil { + hookActions.TerminateWithError(err) + return + } + for _, extension := range extensions { + hookActions.SendExtensionData(extension) + } + } + + if err != nil && err != datatransfer.ErrPause { + hookActions.TerminateWithError(err) + } +} + +// gsIncomingResponseHook is a graphsync.OnIncomingResponseHook. We use it to pass on responses +func (t *Transport) gsIncomingResponseHook(p peer.ID, response graphsync.ResponseData, hookActions graphsync.IncomingResponseHookActions) { + chid, ok := t.requestIDToChannelID.load(response.RequestID()) + if !ok { + return + } + + responseMessage, err := t.processExtension(chid, response, p, incomingReqExtensions) + + if responseMessage != nil { + extensions, extensionErr := extension.ToExtensionData(responseMessage, t.supportedExtensions) + if extensionErr != nil { + hookActions.TerminateWithError(err) + return + } + for _, extension := range extensions { + hookActions.UpdateRequestWithExtensions(extension) + } + } + + if err != nil { + hookActions.TerminateWithError(err) + } + + // In a case where the transfer sends blocks immediately this extension may contain both a + // response message and a revalidation request so we trigger OnResponseReceived again for this + // specific extension name + _, err = t.processExtension(chid, response, p, []graphsync.ExtensionName{extension.ExtensionOutgoingBlock1_1}) + + if err != nil { + hookActions.TerminateWithError(err) + } +} + +func (t *Transport) processExtension(chid datatransfer.ChannelID, gsMsg extension.GsExtended, p peer.ID, exts []graphsync.ExtensionName) (datatransfer.Message, error) { + // if this is a push request the sender is us. + msg, err := extension.GetTransferData(gsMsg, exts) + if err != nil { + return nil, err + } + + // extension not found; probably not our request. + if msg == nil { + return nil, nil + } + + if msg.IsRequest() { + + // only accept request message updates when original message was also request + if (chid != datatransfer.ChannelID{ID: msg.TransferID(), Initiator: p, Responder: t.peerID}) { + return nil, errors.New("received request on response channel") + } + dtRequest := msg.(datatransfer.Request) + return t.events.OnRequestReceived(chid, dtRequest) + } + + // only accept response message updates when original message was also response + if (chid != datatransfer.ChannelID{ID: msg.TransferID(), Initiator: t.peerID, Responder: p}) { + return nil, errors.New("received response on request channel") + } + + dtResponse := msg.(datatransfer.Response) + return nil, t.events.OnResponseReceived(chid, dtResponse) +} + +func (t *Transport) gsRequestorCancelledListener(p peer.ID, request graphsync.RequestData) { + chid, ok := t.requestIDToChannelID.load(request.ID()) + if !ok { + return + } + + ch, err := t.getDTChannel(chid) + if err != nil { + if !errors.Is(datatransfer.ErrChannelNotFound, err) { + log.Errorf("requestor cancelled: getting channel %s: %s", chid, err) + } + return + } + + log.Debugf("%s: requester cancelled data-transfer", chid) + ch.onRequesterCancelled() +} + +// Called when there is a graphsync error sending data +func (t *Transport) gsNetworkSendErrorListener(p peer.ID, request graphsync.RequestData, gserr error) { + // Fire an error if the graphsync request was made by this node or the remote peer + chid, ok := t.requestIDToChannelID.load(request.ID()) + if !ok { + return + } + + err := t.events.OnSendDataError(chid, gserr) + if err != nil { + log.Errorf("failed to fire transport send error %s: %s", gserr, err) + } +} + +// Called when there is a graphsync error receiving data +func (t *Transport) gsNetworkReceiveErrorListener(p peer.ID, gserr error) { + // Fire a receive data error on all ongoing graphsync transfers with that + // peer + t.requestIDToChannelID.forEach(func(k graphsync.RequestID, sending bool, chid datatransfer.ChannelID) { + if chid.Initiator != p && chid.Responder != p { + return + } + + err := t.events.OnReceiveDataError(chid, gserr) + if err != nil { + log.Errorf("failed to fire transport receive error %s: %s", gserr, err) + } + }) +} + +func (t *Transport) newDTChannel(chid datatransfer.ChannelID) *dtChannel { + return &dtChannel{ + t: t, + channelID: chid, + opened: make(chan graphsync.RequestID, 1), + } +} + +func (t *Transport) trackDTChannel(chid datatransfer.ChannelID) *dtChannel { + t.dtChannelsLk.Lock() + defer t.dtChannelsLk.Unlock() + + ch, ok := t.dtChannels[chid] + if !ok { + ch = t.newDTChannel(chid) + t.dtChannels[chid] = ch + } + + return ch +} + +func (t *Transport) getDTChannel(chid datatransfer.ChannelID) (*dtChannel, error) { + if t.events == nil { + return nil, datatransfer.ErrHandlerNotSet + } + + t.dtChannelsLk.RLock() + defer t.dtChannelsLk.RUnlock() + + ch, ok := t.dtChannels[chid] + if !ok { + return nil, fmt.Errorf("channel %s: %w", chid, datatransfer.ErrChannelNotFound) + } + return ch, nil +} + +// Info needed to keep track of a data transfer channel +type dtChannel struct { + channelID datatransfer.ChannelID + t *Transport + + lk sync.RWMutex + isOpen bool + requestID *graphsync.RequestID + completed chan struct{} + requesterCancelled bool + xferStarted bool + pendingExtensions []graphsync.ExtensionData + + opened chan graphsync.RequestID + + storeLk sync.RWMutex + storeRegistered bool +} + +// Info needed to monitor an ongoing graphsync request +type gsReq struct { + channelID datatransfer.ChannelID + responseChan <-chan graphsync.ResponseProgress + errChan <-chan error + onComplete func() +} + +// Open a graphsync request for data to the remote peer +func (c *dtChannel) open( + ctx context.Context, + chid datatransfer.ChannelID, + dataSender peer.ID, + root ipld.Link, + stor ipld.Node, + channel datatransfer.ChannelState, + exts []graphsync.ExtensionData, +) (*gsReq, error) { + c.lk.Lock() + defer c.lk.Unlock() + + // If there is an existing graphsync request for this channelID + if c.requestID != nil { + // Cancel the existing graphsync request + completed := c.completed + errch := c.cancel(ctx) + + // Wait for the complete callback to be called + err := waitForCompleteHook(ctx, completed) + if err != nil { + return nil, fmt.Errorf("%s: waiting for cancelled graphsync request to complete: %w", chid, err) + } + + // Wait for the cancel request method to complete + select { + case err = <-errch: + case <-ctx.Done(): + err = fmt.Errorf("timed out waiting for graphsync request to be cancelled") + } + if err != nil { + return nil, fmt.Errorf("%s: restarting graphsync request: %w", chid, err) + } + } + + // Set up a completed channel that will be closed when the request + // completes (or is cancelled) + completed := make(chan struct{}) + var onCompleteOnce sync.Once + onComplete := func() { + // Ensure the channel is only closed once + onCompleteOnce.Do(func() { + log.Debugw("closing the completion ch for data-transfer channel", "chid", chid) + close(completed) + }) + } + c.completed = completed + + // Open a new graphsync request + msg := fmt.Sprintf("Opening graphsync request to %s for root %s", dataSender, root) + if channel != nil { + msg += fmt.Sprintf(" with %d Blocks already received", channel.ReceivedCidsTotal()) + } + log.Info(msg) + responseChan, errChan := c.t.gs.Request(ctx, dataSender, root, stor, exts...) + + // Wait for graphsync "request opened" callback + select { + case <-ctx.Done(): + return nil, ctx.Err() + case requestID := <-c.opened: + // Mark the channel as open and save the Graphsync request key + c.isOpen = true + c.requestID = &requestID + } + + return &gsReq{ + channelID: chid, + responseChan: responseChan, + errChan: errChan, + onComplete: onComplete, + }, nil +} + +func waitForCompleteHook(ctx context.Context, completed chan struct{}) error { + // Wait for the cancel to propagate through to graphsync, and for + // the graphsync request to complete + select { + case <-completed: + return nil + case <-time.After(maxGSCancelWait): + // Fail-safe: give up waiting after a certain amount of time + return nil + case <-ctx.Done(): + return ctx.Err() + } +} + +// gsReqOpened is called when graphsync makes a request to the remote peer to ask for data +func (c *dtChannel) gsReqOpened(requestID graphsync.RequestID, hookActions graphsync.OutgoingRequestHookActions) { + // Tell graphsync to store the received blocks in the registered store + if c.hasStore() { + hookActions.UsePersistenceOption("data-transfer-" + c.channelID.String()) + } + log.Infow("outgoing graphsync request", "peer", c.channelID.OtherParty(c.t.peerID), "graphsync request id", requestID, "data transfer channel id", c.channelID) + // Save a mapping from the graphsync key to the channel ID so that + // subsequent graphsync callbacks are associated with this channel + c.t.requestIDToChannelID.set(requestID, false, c.channelID) + + c.opened <- requestID +} + +// gsDataRequestRcvd is called when the transport receives an incoming request +// for data. +// Note: Must be called under the lock. +func (c *dtChannel) gsDataRequestRcvd(requestID graphsync.RequestID, hookActions graphsync.IncomingRequestHookActions) { + log.Debugf("%s: received request for data, req_id=%d", c.channelID, requestID) + + // If the requester had previously cancelled their request, send any + // message that was queued since the cancel + if c.requesterCancelled { + c.requesterCancelled = false + + extensions := c.pendingExtensions + c.pendingExtensions = nil + for _, ext := range extensions { + hookActions.SendExtensionData(ext) + } + } + + // Tell graphsync to load blocks from the registered store + if c.hasStore() { + hookActions.UsePersistenceOption("data-transfer-" + c.channelID.String()) + } + + // Save a mapping from the graphsync key to the channel ID so that + // subsequent graphsync callbacks are associated with this channel + c.requestID = &requestID + log.Infow("incoming graphsync request", "peer", c.channelID.OtherParty(c.t.peerID), "graphsync request id", requestID, "data transfer channel id", c.channelID) + c.t.requestIDToChannelID.set(requestID, true, c.channelID) + + c.isOpen = true +} + +func (c *dtChannel) pause(ctx context.Context) error { + c.lk.Lock() + defer c.lk.Unlock() + + // Check if the channel was already cancelled + if c.requestID == nil { + log.Debugf("%s: channel was cancelled so not pausing channel", c.channelID) + return nil + } + + // If the requester cancelled, bail out + if c.requesterCancelled { + log.Debugf("%s: requester has cancelled so not pausing response", c.channelID) + return nil + } + + // Pause the response + log.Debugf("%s: pausing response", c.channelID) + return c.t.gs.Pause(ctx, *c.requestID) +} + +func (c *dtChannel) resume(ctx context.Context, msg datatransfer.Message) error { + c.lk.Lock() + defer c.lk.Unlock() + + // Check if the channel was already cancelled + if c.requestID == nil { + log.Debugf("%s: channel was cancelled so not resuming channel", c.channelID) + return nil + } + + var extensions []graphsync.ExtensionData + if msg != nil { + var err error + extensions, err = extension.ToExtensionData(msg, c.t.supportedExtensions) + if err != nil { + return err + } + } + + // If the requester cancelled, bail out + if c.requesterCancelled { + // If there was an associated message, we still want to send it to the + // remote peer. We're not sending any message now, so instead queue up + // the message to be sent next time the peer makes a request to us. + c.pendingExtensions = append(c.pendingExtensions, extensions...) + + log.Debugf("%s: requester has cancelled so not unpausing response", c.channelID) + return nil + } + + // Record that the transfer has started + c.xferStarted = true + + log.Debugf("%s: unpausing response", c.channelID) + return c.t.gs.Unpause(ctx, *c.requestID, extensions...) +} + +func (c *dtChannel) close(ctx context.Context) error { + var errch chan error + c.lk.Lock() + { + // Check if the channel was already cancelled + if c.requestID != nil { + errch = c.cancel(ctx) + } + } + c.lk.Unlock() + + // Wait for the cancel message to complete + select { + case err := <-errch: + return err + case <-ctx.Done(): + return ctx.Err() + } +} + +// Called when the responder gets a cancel message from the requester +func (c *dtChannel) onRequesterCancelled() { + c.lk.Lock() + defer c.lk.Unlock() + + c.requesterCancelled = true +} + +func (c *dtChannel) hasStore() bool { + c.storeLk.RLock() + defer c.storeLk.RUnlock() + + return c.storeRegistered +} + +// Use the given loader and storer to get / put blocks for the data-transfer. +// Note that each data-transfer channel uses a separate blockstore. +func (c *dtChannel) useStore(lsys ipld.LinkSystem) error { + c.storeLk.Lock() + defer c.storeLk.Unlock() + + // Register the channel's store with graphsync + err := c.t.gs.RegisterPersistenceOption("data-transfer-"+c.channelID.String(), lsys) + if err != nil { + return err + } + + c.storeRegistered = true + + return nil +} + +func (c *dtChannel) cleanup() { + c.lk.Lock() + defer c.lk.Unlock() + + log.Debugf("%s: cleaning up channel", c.channelID) + + if c.hasStore() { + // Unregister the channel's store from graphsync + opt := "data-transfer-" + c.channelID.String() + err := c.t.gs.UnregisterPersistenceOption(opt) + if err != nil { + log.Errorf("failed to unregister persistence option %s: %s", opt, err) + } + } + + // Clean up mapping from gs key to channel ID + c.t.requestIDToChannelID.deleteRefs(c.channelID) +} + +func (c *dtChannel) shutdown(ctx context.Context) error { + // Cancel the graphsync request + c.lk.Lock() + errch := c.cancel(ctx) + c.lk.Unlock() + + // Wait for the cancel message to complete + select { + case err := <-errch: + return err + case <-ctx.Done(): + return ctx.Err() + } +} + +// Cancel the graphsync request. +// Note: must be called under the lock. +func (c *dtChannel) cancel(ctx context.Context) chan error { + errch := make(chan error, 1) + + // Check that the request has not already been cancelled + if c.requesterCancelled || c.requestID == nil { + errch <- nil + return errch + } + + // Clear the graphsync key to indicate that the request has been cancelled + requestID := c.requestID + c.requestID = nil + + go func() { + log.Debugf("%s: cancelling request", c.channelID) + err := c.t.gs.Cancel(ctx, *requestID) + + // Ignore "request not found" errors + if err != nil && !errors.Is(graphsync.RequestNotFoundErr{}, err) { + errch <- fmt.Errorf("cancelling graphsync request for channel %s: %w", c.channelID, err) + } else { + errch <- nil + } + }() + + return errch +} + +type channelInfo struct { + sending bool + channelID datatransfer.ChannelID +} + +// Used in graphsync callbacks to map from graphsync request to the +// associated data-transfer channel ID. +type requestIDToChannelIDMap struct { + lk sync.RWMutex + m map[graphsync.RequestID]channelInfo +} + +func newRequestIDToChannelIDMap() *requestIDToChannelIDMap { + return &requestIDToChannelIDMap{ + m: make(map[graphsync.RequestID]channelInfo), + } +} + +// get the value for a key +func (m *requestIDToChannelIDMap) load(key graphsync.RequestID) (datatransfer.ChannelID, bool) { + m.lk.RLock() + defer m.lk.RUnlock() + + val, ok := m.m[key] + return val.channelID, ok +} + +// set the value for a key +func (m *requestIDToChannelIDMap) set(key graphsync.RequestID, sending bool, chid datatransfer.ChannelID) { + m.lk.Lock() + defer m.lk.Unlock() + + m.m[key] = channelInfo{sending, chid} +} + +// call f for each key / value in the map +func (m *requestIDToChannelIDMap) forEach(f func(k graphsync.RequestID, isSending bool, chid datatransfer.ChannelID)) { + m.lk.RLock() + defer m.lk.RUnlock() + + for k, ch := range m.m { + f(k, ch.sending, ch.channelID) + } +} + +// delete any keys that reference this value +func (m *requestIDToChannelIDMap) deleteRefs(id datatransfer.ChannelID) { + m.lk.Lock() + defer m.lk.Unlock() + + for k, ch := range m.m { + if ch.channelID == id { + delete(m.m, k) + } + } +} diff --git a/datatransfer/types.go b/datatransfer/types.go new file mode 100644 index 000000000..6c9c6504c --- /dev/null +++ b/datatransfer/types.go @@ -0,0 +1,428 @@ +package datatransfer + +import ( + "fmt" + "time" + + "github.com/filecoin-project/boost/datatransfer/encoding" + "github.com/filecoin-project/go-statemachine/fsm" + "github.com/ipfs/go-cid" + "github.com/ipld/go-ipld-prime" + "github.com/ipld/go-ipld-prime/datamodel" + "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" +) + +//go:generate cbor-gen-for ChannelID ChannelStages ChannelStage Log + +// TypeIdentifier is a unique string identifier for a type of encodable object in a +// registry +type TypeIdentifier string + +// EmptyTypeIdentifier means there is no voucher present +const EmptyTypeIdentifier = TypeIdentifier("") + +// Voucher is used to validate +// a data transfer request against the underlying storage or retrieval deal +// that precipitated it. The only requirement is a voucher can read and write +// from bytes, and has a string identifier type +type Voucher Registerable + +// VoucherResult is used to provide option additional information about a +// voucher being rejected or accepted +type VoucherResult Registerable + +// TransferID is an identifier for a data transfer, shared between +// request/responder and unique to the requester +type TransferID uint64 + +// TypedVoucher is a voucher or voucher result in IPLD form and an associated +// type identifier for that voucher or voucher result +type TypedVoucher struct { + Voucher datamodel.Node + Type TypeIdentifier +} + +// Registerable is a type of object in a registry. It must be encodable and must +// have a single method that uniquely identifies its type +type Registerable interface { + encoding.Encodable + // Type is a unique string identifier for this voucher type + Type() TypeIdentifier +} + +// Equals is a utility to compare that two TypedVouchers are the same - both type +// and the voucher's IPLD content +func (tv1 TypedVoucher) Equals(tv2 TypedVoucher) bool { + return tv1.Type == tv2.Type && ipld.DeepEqual(tv1.Voucher, tv2.Voucher) +} + +// ChannelID is a unique identifier for a channel, distinct by both the other +// party's peer ID + the transfer ID +type ChannelID struct { + Initiator peer.ID + Responder peer.ID + ID TransferID +} + +func (c ChannelID) String() string { + return fmt.Sprintf("%s-%s-%d", c.Initiator, c.Responder, c.ID) +} + +// OtherParty returns the peer on the other side of the request, depending +// on whether this peer is the initiator or responder +func (c ChannelID) OtherParty(thisPeer peer.ID) peer.ID { + if thisPeer == c.Initiator { + return c.Responder + } + return c.Initiator +} + +// Channel represents all the parameters for a single data transfer +type Channel interface { + // TransferID returns the transfer id for this channel + TransferID() TransferID + + // BaseCID returns the CID that is at the root of this data transfer + BaseCID() cid.Cid + + // Selector returns the IPLD selector for this data transfer (represented as + // an IPLD node) + Selector() datamodel.Node + + // Voucher returns the initial voucher for this data transfer + Voucher() Voucher + + // Sender returns the peer id for the node that is sending data + Sender() peer.ID + + // Recipient returns the peer id for the node that is receiving data + Recipient() peer.ID + + // TotalSize returns the total size for the data being transferred + TotalSize() uint64 + + // IsPull returns whether this is a pull request + IsPull() bool + + // ChannelID returns the ChannelID for this request + ChannelID() ChannelID + + // OtherPeer returns the counter party peer for this channel + OtherPeer() peer.ID +} + +// ChannelState is channel parameters plus it's current state +type ChannelState interface { + Channel + + // SelfPeer returns the peer this channel belongs to + SelfPeer() peer.ID + + // Status is the current status of this channel + Status() Status + + // Sent returns the number of bytes sent + Sent() uint64 + + // Received returns the number of bytes received + Received() uint64 + + // Message offers additional information about the current status + Message() string + + // Vouchers returns all vouchers sent on this channel + Vouchers() []Voucher + + // VoucherResults are results of vouchers sent on the channel + VoucherResults() []VoucherResult + + // LastVoucher returns the last voucher sent on the channel + LastVoucher() Voucher + + // LastVoucherResult returns the last voucher result sent on the channel + LastVoucherResult() VoucherResult + + // ReceivedCidsTotal returns the number of (non-unique) cids received so far + // on the channel - note that a block can exist in more than one place in the DAG + ReceivedCidsTotal() int64 + + // QueuedCidsTotal returns the number of (non-unique) cids queued so far + // on the channel - note that a block can exist in more than one place in the DAG + QueuedCidsTotal() int64 + + // SentCidsTotal returns the number of (non-unique) cids sent so far + // on the channel - note that a block can exist in more than one place in the DAG + SentCidsTotal() int64 + + // Queued returns the number of bytes read from the node and queued for sending + Queued() uint64 + + // Stages returns the timeline of events this data transfer has gone through, + // for observability purposes. + // + // It is unsafe for the caller to modify the return value, and changes + // may not be persisted. It should be treated as immutable. + Stages() *ChannelStages +} + +// ChannelStages captures a timeline of the progress of a data transfer channel, +// grouped by stages. +// +// EXPERIMENTAL; subject to change. +type ChannelStages struct { + // Stages contains an entry for every stage the channel has gone through. + // Each stage then contains logs. + Stages []*ChannelStage +} + +// ChannelStage traces the execution of a data transfer channel stage. +// +// EXPERIMENTAL; subject to change. +type ChannelStage struct { + // Human-readable fields. + // TODO: these _will_ need to be converted to canonical representations, so + // they are machine readable. + Name string + Description string + + // Timestamps. + // TODO: may be worth adding an exit timestamp. It _could_ be inferred from + // the start of the next stage, or from the timestamp of the last log line + // if this is a terminal stage. But that's non-determistic and it relies on + // assumptions. + CreatedTime cbg.CborTime + UpdatedTime cbg.CborTime + + // Logs contains a detailed timeline of events that occurred inside + // this stage. + Logs []*Log +} + +// Log represents a point-in-time event that occurred inside a channel stage. +// +// EXPERIMENTAL; subject to change. +type Log struct { + // Log is a human readable message. + // + // TODO: this _may_ need to be converted to a canonical data model so it + // is machine-readable. + Log string + + UpdatedTime cbg.CborTime +} + +// AddLog adds a log to the specified stage, creating the stage if +// it doesn't exist yet. +// +// EXPERIMENTAL; subject to change. +func (cs *ChannelStages) AddLog(stage, msg string) { + if cs == nil { + return + } + + now := curTime() + st := cs.GetStage(stage) + if st == nil { + st = &ChannelStage{ + CreatedTime: now, + } + cs.Stages = append(cs.Stages, st) + } + + st.Name = stage + st.UpdatedTime = now + if msg != "" && (len(st.Logs) == 0 || st.Logs[len(st.Logs)-1].Log != msg) { + // only add the log if it's not a duplicate. + st.Logs = append(st.Logs, &Log{msg, now}) + } +} + +// GetStage returns the ChannelStage object for a named stage, or nil if not found. +// +// TODO: the input should be a strongly-typed enum instead of a free-form string. +// TODO: drop Get from GetStage to make this code more idiomatic. Return a +// +// second ok boolean to make it even more idiomatic. +// +// EXPERIMENTAL; subject to change. +func (cs *ChannelStages) GetStage(stage string) *ChannelStage { + if cs == nil { + return nil + } + + for _, s := range cs.Stages { + if s.Name == stage { + return s + } + } + + return nil +} + +func curTime() cbg.CborTime { + now := time.Now() + return cbg.CborTime(time.Unix(0, now.UnixNano()).UTC()) +} + +// Status is the status of transfer for a given channel +type Status uint64 + +const ( + // Requested means a data transfer was requested by has not yet been approved + Requested Status = iota + + // Ongoing means the data transfer is in progress + Ongoing + + // TransferFinished indicates the initiator is done sending/receiving + // data but is awaiting confirmation from the responder + TransferFinished + + // ResponderCompleted indicates the initiator received a message from the + // responder that it's completed + ResponderCompleted + + // Finalizing means the responder is awaiting a final message from the initator to + // consider the transfer done + Finalizing + + // Completing just means we have some final cleanup for a completed request + Completing + + // Completed means the data transfer is completed successfully + Completed + + // Failing just means we have some final cleanup for a failed request + Failing + + // Failed means the data transfer failed + Failed + + // Cancelling just means we have some final cleanup for a cancelled request + Cancelling + + // Cancelled means the data transfer ended prematurely + Cancelled + + // DEPRECATED: Use InitiatorPaused() method on ChannelState + InitiatorPaused + + // DEPRECATED: Use ResponderPaused() method on ChannelState + ResponderPaused + + // DEPRECATED: Use BothPaused() method on ChannelState + BothPaused + + // ResponderFinalizing is a unique state where the responder is awaiting a final voucher + ResponderFinalizing + + // ResponderFinalizingTransferFinished is a unique state where the responder is awaiting a final voucher + // and we have received all data + ResponderFinalizingTransferFinished + + // ChannelNotFoundError means the searched for data transfer does not exist + ChannelNotFoundError + + // Queued indicates a data transfer request has been accepted, but is not actively transfering yet + Queued + + // AwaitingAcceptance indicates a transfer request is actively being processed by the transport + // even if the remote has not yet responded that it's accepted the transfer. Such a state can + // occur, for example, in a requestor-initiated transfer that starts processing prior to receiving + // acceptance from the server. + AwaitingAcceptance +) + +type statusList []Status + +func (sl statusList) Contains(s Status) bool { + for _, ts := range sl { + if ts == s { + return true + } + } + return false +} + +func (sl statusList) AsFSMStates() []fsm.StateKey { + sk := make([]fsm.StateKey, 0, len(sl)) + for _, s := range sl { + sk = append(sk, s) + } + return sk +} + +var NotAcceptedStates = statusList{ + Requested, + AwaitingAcceptance, + Cancelled, + Cancelling, + Failed, + Failing, + ChannelNotFoundError} + +func (s Status) IsAccepted() bool { + return !NotAcceptedStates.Contains(s) +} +func (s Status) String() string { + return Statuses[s] +} + +var FinalizationStatuses = statusList{Finalizing, Completed, Completing} + +func (s Status) InFinalization() bool { + return FinalizationStatuses.Contains(s) +} + +var TransferCompleteStates = statusList{ + TransferFinished, + ResponderFinalizingTransferFinished, + Finalizing, + Completed, + Completing, + Failing, + Failed, + Cancelling, + Cancelled, + ChannelNotFoundError, +} + +func (s Status) TransferComplete() bool { + return TransferCompleteStates.Contains(s) +} + +var TransferringStates = statusList{ + Ongoing, + ResponderCompleted, + ResponderFinalizing, + AwaitingAcceptance, +} + +func (s Status) Transferring() bool { + return TransferringStates.Contains(s) +} + +// Statuses are human readable names for data transfer states +var Statuses = map[Status]string{ + // Requested means a data transfer was requested by has not yet been approved + Requested: "Requested", + Ongoing: "Ongoing", + TransferFinished: "TransferFinished", + ResponderCompleted: "ResponderCompleted", + Finalizing: "Finalizing", + Completing: "Completing", + Completed: "Completed", + Failing: "Failing", + Failed: "Failed", + Cancelling: "Cancelling", + Cancelled: "Cancelled", + InitiatorPaused: "InitiatorPaused", + ResponderPaused: "ResponderPaused", + BothPaused: "BothPaused", + ResponderFinalizing: "ResponderFinalizing", + ResponderFinalizingTransferFinished: "ResponderFinalizingTransferFinished", + ChannelNotFoundError: "ChannelNotFoundError", + Queued: "Queued", + AwaitingAcceptance: "AwaitingAcceptance", +} diff --git a/datatransfer/types_cbor_gen.go b/datatransfer/types_cbor_gen.go new file mode 100644 index 000000000..e212d360b --- /dev/null +++ b/datatransfer/types_cbor_gen.go @@ -0,0 +1,487 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package datatransfer + +import ( + "fmt" + "io" + "math" + "sort" + + cid "github.com/ipfs/go-cid" + peer "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + xerrors "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +var lengthBufChannelID = []byte{131} + +func (t *ChannelID) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufChannelID); err != nil { + return err + } + + // t.Initiator (peer.ID) (string) + if len(t.Initiator) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Initiator was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Initiator))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Initiator)); err != nil { + return err + } + + // t.Responder (peer.ID) (string) + if len(t.Responder) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Responder was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Responder))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Responder)); err != nil { + return err + } + + // t.ID (datatransfer.TransferID) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.ID)); err != nil { + return err + } + + return nil +} + +func (t *ChannelID) UnmarshalCBOR(r io.Reader) (err error) { + *t = ChannelID{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 3 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Initiator (peer.ID) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Initiator = peer.ID(sval) + } + // t.Responder (peer.ID) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Responder = peer.ID(sval) + } + // t.ID (datatransfer.TransferID) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.ID = TransferID(extra) + + } + return nil +} + +var lengthBufChannelStages = []byte{129} + +func (t *ChannelStages) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufChannelStages); err != nil { + return err + } + + // t.Stages ([]*datatransfer.ChannelStage) (slice) + if len(t.Stages) > cbg.MaxLength { + return xerrors.Errorf("Slice value in field t.Stages was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajArray, uint64(len(t.Stages))); err != nil { + return err + } + for _, v := range t.Stages { + if err := v.MarshalCBOR(cw); err != nil { + return err + } + + } + return nil +} + +func (t *ChannelStages) UnmarshalCBOR(r io.Reader) (err error) { + *t = ChannelStages{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 1 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Stages ([]*datatransfer.ChannelStage) (slice) + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + + if extra > cbg.MaxLength { + return fmt.Errorf("t.Stages: array too large (%d)", extra) + } + + if maj != cbg.MajArray { + return fmt.Errorf("expected cbor array") + } + + if extra > 0 { + t.Stages = make([]*ChannelStage, extra) + } + + for i := 0; i < int(extra); i++ { + { + var maj byte + var extra uint64 + var err error + _ = maj + _ = extra + _ = err + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Stages[i] = new(ChannelStage) + if err := t.Stages[i].UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Stages[i] pointer: %w", err) + } + } + + } + + } + } + return nil +} + +var lengthBufChannelStage = []byte{133} + +func (t *ChannelStage) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufChannelStage); err != nil { + return err + } + + // t.Name (string) (string) + if len(t.Name) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Name was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Name))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Name)); err != nil { + return err + } + + // t.Description (string) (string) + if len(t.Description) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Description was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Description))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Description)); err != nil { + return err + } + + // t.CreatedTime (typegen.CborTime) (struct) + if err := t.CreatedTime.MarshalCBOR(cw); err != nil { + return err + } + + // t.UpdatedTime (typegen.CborTime) (struct) + if err := t.UpdatedTime.MarshalCBOR(cw); err != nil { + return err + } + + // t.Logs ([]*datatransfer.Log) (slice) + if len(t.Logs) > cbg.MaxLength { + return xerrors.Errorf("Slice value in field t.Logs was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajArray, uint64(len(t.Logs))); err != nil { + return err + } + for _, v := range t.Logs { + if err := v.MarshalCBOR(cw); err != nil { + return err + } + + } + return nil +} + +func (t *ChannelStage) UnmarshalCBOR(r io.Reader) (err error) { + *t = ChannelStage{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 5 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Name (string) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Name = string(sval) + } + // t.Description (string) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Description = string(sval) + } + // t.CreatedTime (typegen.CborTime) (struct) + + { + + if err := t.CreatedTime.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.CreatedTime: %w", err) + } + + } + // t.UpdatedTime (typegen.CborTime) (struct) + + { + + if err := t.UpdatedTime.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.UpdatedTime: %w", err) + } + + } + // t.Logs ([]*datatransfer.Log) (slice) + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + + if extra > cbg.MaxLength { + return fmt.Errorf("t.Logs: array too large (%d)", extra) + } + + if maj != cbg.MajArray { + return fmt.Errorf("expected cbor array") + } + + if extra > 0 { + t.Logs = make([]*Log, extra) + } + + for i := 0; i < int(extra); i++ { + { + var maj byte + var extra uint64 + var err error + _ = maj + _ = extra + _ = err + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Logs[i] = new(Log) + if err := t.Logs[i].UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Logs[i] pointer: %w", err) + } + } + + } + + } + } + return nil +} + +var lengthBufLog = []byte{130} + +func (t *Log) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufLog); err != nil { + return err + } + + // t.Log (string) (string) + if len(t.Log) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Log was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Log))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Log)); err != nil { + return err + } + + // t.UpdatedTime (typegen.CborTime) (struct) + if err := t.UpdatedTime.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *Log) UnmarshalCBOR(r io.Reader) (err error) { + *t = Log{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 2 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Log (string) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Log = string(sval) + } + // t.UpdatedTime (typegen.CborTime) (struct) + + { + + if err := t.UpdatedTime.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.UpdatedTime: %w", err) + } + + } + return nil +} diff --git a/documentation/en/api-v1-methods.md b/documentation/en/api-v1-methods.md index 4b0230cf5..e282b358b 100644 --- a/documentation/en/api-v1-methods.md +++ b/documentation/en/api-v1-methods.md @@ -1,8 +1,6 @@ # Groups * [](#) * [Discover](#discover) -* [Actor](#actor) - * [ActorSectorSize](#actorsectorsize) * [Auth](#auth) * [AuthNew](#authnew) * [AuthVerify](#authverify) @@ -11,14 +9,6 @@ * [BlockstoreGetSize](#blockstoregetsize) * [BlockstoreHas](#blockstorehas) * [Boost](#boost) - * [BoostDagstoreDestroyShard](#boostdagstoredestroyshard) - * [BoostDagstoreGC](#boostdagstoregc) - * [BoostDagstoreInitializeAll](#boostdagstoreinitializeall) - * [BoostDagstoreInitializeShard](#boostdagstoreinitializeshard) - * [BoostDagstoreListShards](#boostdagstorelistshards) - * [BoostDagstorePiecesContainingMultihash](#boostdagstorepiecescontainingmultihash) - * [BoostDagstoreRecoverShard](#boostdagstorerecovershard) - * [BoostDagstoreRegisterShard](#boostdagstoreregistershard) * [BoostDeal](#boostdeal) * [BoostDealBySignedProposalCid](#boostdealbysignedproposalcid) * [BoostDirectDeal](#boostdirectdeal) @@ -33,39 +23,11 @@ * [BoostLegacyDealByProposalCid](#boostlegacydealbyproposalcid) * [BoostMakeDeal](#boostmakedeal) * [BoostOfflineDealWithData](#boostofflinedealwithdata) -* [Deals](#deals) - * [DealsConsiderOfflineRetrievalDeals](#dealsconsiderofflineretrievaldeals) - * [DealsConsiderOfflineStorageDeals](#dealsconsiderofflinestoragedeals) - * [DealsConsiderOnlineRetrievalDeals](#dealsconsideronlineretrievaldeals) - * [DealsConsiderOnlineStorageDeals](#dealsconsideronlinestoragedeals) - * [DealsConsiderUnverifiedStorageDeals](#dealsconsiderunverifiedstoragedeals) - * [DealsConsiderVerifiedStorageDeals](#dealsconsiderverifiedstoragedeals) - * [DealsPieceCidBlocklist](#dealspiececidblocklist) - * [DealsSetConsiderOfflineRetrievalDeals](#dealssetconsiderofflineretrievaldeals) - * [DealsSetConsiderOfflineStorageDeals](#dealssetconsiderofflinestoragedeals) - * [DealsSetConsiderOnlineRetrievalDeals](#dealssetconsideronlineretrievaldeals) - * [DealsSetConsiderOnlineStorageDeals](#dealssetconsideronlinestoragedeals) - * [DealsSetConsiderUnverifiedStorageDeals](#dealssetconsiderunverifiedstoragedeals) - * [DealsSetConsiderVerifiedStorageDeals](#dealssetconsiderverifiedstoragedeals) - * [DealsSetPieceCidBlocklist](#dealssetpiececidblocklist) * [I](#i) * [ID](#id) * [Log](#log) * [LogList](#loglist) * [LogSetLevel](#logsetlevel) -* [Market](#market) - * [MarketCancelDataTransfer](#marketcanceldatatransfer) - * [MarketDataTransferUpdates](#marketdatatransferupdates) - * [MarketGetAsk](#marketgetask) - * [MarketGetRetrievalAsk](#marketgetretrievalask) - * [MarketImportDealData](#marketimportdealdata) - * [MarketListDataTransfers](#marketlistdatatransfers) - * [MarketListIncompleteDeals](#marketlistincompletedeals) - * [MarketListRetrievalDeals](#marketlistretrievaldeals) - * [MarketPendingDeals](#marketpendingdeals) - * [MarketRestartDataTransfer](#marketrestartdatatransfer) - * [MarketSetAsk](#marketsetask) - * [MarketSetRetrievalAsk](#marketsetretrievalask) * [Net](#net) * [NetAddrsListen](#netaddrslisten) * [NetAgentVersion](#netagentversion) @@ -94,10 +56,6 @@ * [OnlineBackup](#onlinebackup) * [Pd](#pd) * [PdBuildIndexForPieceCid](#pdbuildindexforpiececid) -* [Runtime](#runtime) - * [RuntimeSubsystems](#runtimesubsystems) -* [Sectors](#sectors) - * [SectorsRefs](#sectorsrefs) ## @@ -120,23 +78,6 @@ Response: } ``` -## Actor - - -### ActorSectorSize -There are not yet any comments for this method. - -Perms: read - -Inputs: -```json -[ - "f01234" -] -``` - -Response: `34359738368` - ## Auth @@ -229,146 +170,6 @@ Response: `true` ## Boost -### BoostDagstoreDestroyShard - - -Perms: admin - -Inputs: -```json -[ - "string value" -] -``` - -Response: `{}` - -### BoostDagstoreGC - - -Perms: admin - -Inputs: `null` - -Response: -```json -[ - { - "Key": "baga6ea4seaqecmtz7iak33dsfshi627abz4i4665dfuzr3qfs4bmad6dx3iigdq", - "Success": false, - "Error": "\u003cerror\u003e" - } -] -``` - -### BoostDagstoreInitializeAll - - -Perms: admin - -Inputs: -```json -[ - { - "MaxConcurrency": 123, - "IncludeSealed": true - } -] -``` - -Response: -```json -{ - "Key": "string value", - "Event": "string value", - "Success": true, - "Error": "string value", - "Total": 123, - "Current": 123 -} -``` - -### BoostDagstoreInitializeShard - - -Perms: admin - -Inputs: -```json -[ - "string value" -] -``` - -Response: `{}` - -### BoostDagstoreListShards - - -Perms: admin - -Inputs: `null` - -Response: -```json -[ - { - "Key": "baga6ea4seaqecmtz7iak33dsfshi627abz4i4665dfuzr3qfs4bmad6dx3iigdq", - "State": "ShardStateAvailable", - "Error": "\u003cerror\u003e" - } -] -``` - -### BoostDagstorePiecesContainingMultihash - - -Perms: read - -Inputs: -```json -[ - "Bw==" -] -``` - -Response: -```json -[ - { - "/": "bafy2bzacea3wsdh6y3a36tb3skempjoxqpuyompjbmfeyf34fi3uy6uue42v4" - } -] -``` - -### BoostDagstoreRecoverShard - - -Perms: admin - -Inputs: -```json -[ - "string value" -] -``` - -Response: `{}` - -### BoostDagstoreRegisterShard - - -Perms: admin - -Inputs: -```json -[ - "string value" -] -``` - -Response: `{}` - ### BoostDeal @@ -738,7 +539,12 @@ Inputs: ] ``` -Response: `{}` +Response: +```json +{ + "/": "bafy2bzacea3wsdh6y3a36tb3skempjoxqpuyompjbmfeyf34fi3uy6uue42v4" +} +``` ### BoostIndexerListMultihashes @@ -805,8 +611,8 @@ Response: "Miner": "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", "Client": "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", "State": 42, - "PiecePath": ".lotusminer/fstmp123", - "MetadataPath": ".lotusminer/fstmp123", + "PiecePath": "", + "MetadataPath": "", "SlashEpoch": 10101, "FastRetrieval": true, "Message": "string value", @@ -910,657 +716,109 @@ Response: } ``` -## Deals - - -### DealsConsiderOfflineRetrievalDeals - - -Perms: admin - -Inputs: `null` - -Response: `true` - -### DealsConsiderOfflineStorageDeals - - -Perms: admin - -Inputs: `null` - -Response: `true` - -### DealsConsiderOnlineRetrievalDeals - - -Perms: admin - -Inputs: `null` - -Response: `true` - -### DealsConsiderOnlineStorageDeals -There are not yet any comments for this method. - -Perms: admin - -Inputs: `null` +## I -Response: `true` -### DealsConsiderUnverifiedStorageDeals +### ID -Perms: admin +Perms: read Inputs: `null` -Response: `true` - -### DealsConsiderVerifiedStorageDeals - - -Perms: admin +Response: `"12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf"` -Inputs: `null` +## Log -Response: `true` -### DealsPieceCidBlocklist +### LogList -Perms: admin +Perms: write Inputs: `null` Response: ```json [ - { - "/": "bafy2bzacea3wsdh6y3a36tb3skempjoxqpuyompjbmfeyf34fi3uy6uue42v4" - } + "string value" ] ``` -### DealsSetConsiderOfflineRetrievalDeals +### LogSetLevel -Perms: admin +Perms: write Inputs: ```json [ - true + "string value", + "string value" ] ``` Response: `{}` -### DealsSetConsiderOfflineStorageDeals - - -Perms: admin +## Net -Inputs: -```json -[ - true -] -``` -Response: `{}` +### NetAddrsListen -### DealsSetConsiderOnlineRetrievalDeals +Perms: read -Perms: admin +Inputs: `null` -Inputs: +Response: ```json -[ - true -] +{ + "ID": "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", + "Addrs": [ + "/ip4/52.36.61.156/tcp/1347/p2p/12D3KooWFETiESTf1v4PGUvtnxMAcEFMzLZbJGg4tjWfGEimYior" + ] +} ``` -Response: `{}` - -### DealsSetConsiderOnlineStorageDeals +### NetAgentVersion -Perms: admin +Perms: read Inputs: ```json [ - true + "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf" ] ``` -Response: `{}` +Response: `"string value"` + +### NetAutoNatStatus -### DealsSetConsiderUnverifiedStorageDeals +Perms: read -Perms: admin +Inputs: `null` -Inputs: +Response: ```json -[ - true -] +{ + "Reachability": 1, + "PublicAddrs": [ + "string value" + ] +} ``` -Response: `{}` +### NetBandwidthStats -### DealsSetConsiderVerifiedStorageDeals +Perms: read -Perms: admin +Inputs: `null` -Inputs: -```json -[ - true -] -``` - -Response: `{}` - -### DealsSetPieceCidBlocklist - - -Perms: admin - -Inputs: -```json -[ - [ - { - "/": "bafy2bzacea3wsdh6y3a36tb3skempjoxqpuyompjbmfeyf34fi3uy6uue42v4" - } - ] -] -``` - -Response: `{}` - -## I - - -### ID - - -Perms: read - -Inputs: `null` - -Response: `"12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf"` - -## Log - - -### LogList - - -Perms: write - -Inputs: `null` - -Response: -```json -[ - "string value" -] -``` - -### LogSetLevel - - -Perms: write - -Inputs: -```json -[ - "string value", - "string value" -] -``` - -Response: `{}` - -## Market - - -### MarketCancelDataTransfer - - -Perms: write - -Inputs: -```json -[ - 3, - "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", - true -] -``` - -Response: `{}` - -### MarketDataTransferUpdates - - -Perms: write - -Inputs: `null` - -Response: -```json -{ - "TransferID": 3, - "Status": 1, - "BaseCID": { - "/": "bafy2bzacea3wsdh6y3a36tb3skempjoxqpuyompjbmfeyf34fi3uy6uue42v4" - }, - "IsInitiator": true, - "IsSender": true, - "Voucher": "string value", - "Message": "string value", - "OtherPeer": "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", - "Transferred": 42, - "Stages": { - "Stages": [ - { - "Name": "string value", - "Description": "string value", - "CreatedTime": "0001-01-01T00:00:00Z", - "UpdatedTime": "0001-01-01T00:00:00Z", - "Logs": [ - { - "Log": "string value", - "UpdatedTime": "0001-01-01T00:00:00Z" - } - ] - } - ] - } -} -``` - -### MarketGetAsk - - -Perms: read - -Inputs: `null` - -Response: -```json -{ - "Ask": { - "Price": "0", - "VerifiedPrice": "0", - "MinPieceSize": 1032, - "MaxPieceSize": 1032, - "Miner": "f01234", - "Timestamp": 10101, - "Expiry": 10101, - "SeqNo": 42 - }, - "Signature": { - "Type": 2, - "Data": "Ynl0ZSBhcnJheQ==" - } -} -``` - -### MarketGetRetrievalAsk - - -Perms: read - -Inputs: `null` - -Response: -```json -{ - "PricePerByte": "0", - "UnsealPrice": "0", - "PaymentInterval": 42, - "PaymentIntervalIncrease": 42 -} -``` - -### MarketImportDealData - - -Perms: write - -Inputs: -```json -[ - { - "/": "bafy2bzacea3wsdh6y3a36tb3skempjoxqpuyompjbmfeyf34fi3uy6uue42v4" - }, - "string value" -] -``` - -Response: `{}` - -### MarketListDataTransfers - - -Perms: write - -Inputs: `null` - -Response: -```json -[ - { - "TransferID": 3, - "Status": 1, - "BaseCID": { - "/": "bafy2bzacea3wsdh6y3a36tb3skempjoxqpuyompjbmfeyf34fi3uy6uue42v4" - }, - "IsInitiator": true, - "IsSender": true, - "Voucher": "string value", - "Message": "string value", - "OtherPeer": "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", - "Transferred": 42, - "Stages": { - "Stages": [ - { - "Name": "string value", - "Description": "string value", - "CreatedTime": "0001-01-01T00:00:00Z", - "UpdatedTime": "0001-01-01T00:00:00Z", - "Logs": [ - { - "Log": "string value", - "UpdatedTime": "0001-01-01T00:00:00Z" - } - ] - } - ] - } - } -] -``` - -### MarketListIncompleteDeals - - -Perms: read - -Inputs: `null` - -Response: -```json -[ - { - "Proposal": { - "PieceCID": { - "/": "bafy2bzacea3wsdh6y3a36tb3skempjoxqpuyompjbmfeyf34fi3uy6uue42v4" - }, - "PieceSize": 1032, - "VerifiedDeal": true, - "Client": "f01234", - "Provider": "f01234", - "Label": "", - "StartEpoch": 10101, - "EndEpoch": 10101, - "StoragePricePerEpoch": "0", - "ProviderCollateral": "0", - "ClientCollateral": "0" - }, - "ClientSignature": { - "Type": 2, - "Data": "Ynl0ZSBhcnJheQ==" - }, - "ProposalCid": { - "/": "bafy2bzacea3wsdh6y3a36tb3skempjoxqpuyompjbmfeyf34fi3uy6uue42v4" - }, - "AddFundsCid": null, - "PublishCid": null, - "Miner": "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", - "Client": "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", - "State": 42, - "PiecePath": ".lotusminer/fstmp123", - "MetadataPath": ".lotusminer/fstmp123", - "SlashEpoch": 10101, - "FastRetrieval": true, - "Message": "string value", - "FundsReserved": "0", - "Ref": { - "TransferType": "string value", - "Root": { - "/": "bafy2bzacea3wsdh6y3a36tb3skempjoxqpuyompjbmfeyf34fi3uy6uue42v4" - }, - "PieceCid": null, - "PieceSize": 1024, - "RawBlockSize": 42 - }, - "AvailableForRetrieval": true, - "DealID": 5432, - "CreationTime": "0001-01-01T00:00:00Z", - "TransferChannelId": { - "Initiator": "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", - "Responder": "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", - "ID": 3 - }, - "SectorNumber": 9, - "InboundCAR": "string value" - } -] -``` - -### MarketListRetrievalDeals -There are not yet any comments for this method. - -Perms: read - -Inputs: `null` - -Response: -```json -[ - { - "PayloadCID": { - "/": "bafy2bzacea3wsdh6y3a36tb3skempjoxqpuyompjbmfeyf34fi3uy6uue42v4" - }, - "ID": 5, - "Selector": { - "Raw": "Ynl0ZSBhcnJheQ==" - }, - "PieceCID": null, - "PricePerByte": "0", - "PaymentInterval": 42, - "PaymentIntervalIncrease": 42, - "UnsealPrice": "0", - "StoreID": 42, - "ChannelID": { - "Initiator": "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", - "Responder": "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", - "ID": 3 - }, - "PieceInfo": { - "PieceCID": { - "/": "bafy2bzacea3wsdh6y3a36tb3skempjoxqpuyompjbmfeyf34fi3uy6uue42v4" - }, - "Deals": [ - { - "DealID": 5432, - "SectorID": 9, - "Offset": 1032, - "Length": 1032 - } - ] - }, - "Status": 0, - "Receiver": "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", - "TotalSent": 42, - "FundsReceived": "0", - "Message": "string value", - "CurrentInterval": 42, - "LegacyProtocol": true - } -] -``` - -### MarketPendingDeals - - -Perms: write - -Inputs: `null` - -Response: -```json -{ - "Deals": [ - { - "Proposal": { - "PieceCID": { - "/": "bafy2bzacea3wsdh6y3a36tb3skempjoxqpuyompjbmfeyf34fi3uy6uue42v4" - }, - "PieceSize": 1032, - "VerifiedDeal": true, - "Client": "f01234", - "Provider": "f01234", - "Label": "", - "StartEpoch": 10101, - "EndEpoch": 10101, - "StoragePricePerEpoch": "0", - "ProviderCollateral": "0", - "ClientCollateral": "0" - }, - "ClientSignature": { - "Type": 2, - "Data": "Ynl0ZSBhcnJheQ==" - } - } - ], - "PublishPeriodStart": "0001-01-01T00:00:00Z", - "PublishPeriod": 60000000000 -} -``` - -### MarketRestartDataTransfer - - -Perms: write - -Inputs: -```json -[ - 3, - "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", - true -] -``` - -Response: `{}` - -### MarketSetAsk - - -Perms: admin - -Inputs: -```json -[ - "0", - "0", - 10101, - 1032, - 1032 -] -``` - -Response: `{}` - -### MarketSetRetrievalAsk - - -Perms: admin - -Inputs: -```json -[ - { - "PricePerByte": "0", - "UnsealPrice": "0", - "PaymentInterval": 42, - "PaymentIntervalIncrease": 42 - } -] -``` - -Response: `{}` - -## Net - - -### NetAddrsListen - - -Perms: read - -Inputs: `null` - -Response: -```json -{ - "ID": "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf", - "Addrs": [ - "/ip4/52.36.61.156/tcp/1347/p2p/12D3KooWFETiESTf1v4PGUvtnxMAcEFMzLZbJGg4tjWfGEimYior" - ] -} -``` - -### NetAgentVersion - - -Perms: read - -Inputs: -```json -[ - "12D3KooWGzxzKZYveHXtpG6AsrUJBcWxHBFS2HsEoGTxrMLvKXtf" -] -``` - -Response: `"string value"` - -### NetAutoNatStatus - - -Perms: read - -Inputs: `null` - -Response: -```json -{ - "Reachability": 1, - "PublicAddrs": [ - "string value" - ] -} -``` - -### NetBandwidthStats - - -Perms: read - -Inputs: `null` - -Response: +Response: ```json { "TotalIn": 9, @@ -2042,45 +1300,3 @@ Inputs: Response: `{}` -## Runtime - - -### RuntimeSubsystems -RuntimeSubsystems returns the subsystems that are enabled -in this instance. - - -Perms: read - -Inputs: `null` - -Response: -```json -[ - "Markets" -] -``` - -## Sectors - - -### SectorsRefs - - -Perms: read - -Inputs: `null` - -Response: -```json -{ - "98000": [ - { - "SectorID": 100, - "Offset": 10485760, - "Size": 1048576 - } - ] -} -``` - diff --git a/fundmanager/fundmanager.go b/fundmanager/fundmanager.go index 35ff1f3bc..2375eb30e 100644 --- a/fundmanager/fundmanager.go +++ b/fundmanager/fundmanager.go @@ -6,8 +6,8 @@ import ( "fmt" "time" - "github.com/filecoin-project/boost-gfm/storagemarket" "github.com/filecoin-project/boost/db" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" "github.com/filecoin-project/go-address" "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/go-state-types/big" @@ -220,10 +220,10 @@ func (m *FundManager) MoveFundsToEscrow(ctx context.Context, amt abi.TokenAmount // BalanceMarket returns available and locked amounts in escrow // (on chain with the Storage Market Actor) -func (m *FundManager) BalanceMarket(ctx context.Context) (storagemarket.Balance, error) { +func (m *FundManager) BalanceMarket(ctx context.Context) (legacytypes.Balance, error) { bal, err := m.api.StateMarketBalance(ctx, m.cfg.StorageMiner, types.EmptyTSK) if err != nil { - return storagemarket.Balance{}, err + return legacytypes.Balance{}, err } return toSharedBalance(bal), nil @@ -249,8 +249,8 @@ func (m *FundManager) AddressPublishMsg() address.Address { return m.cfg.PubMsgWallet } -func toSharedBalance(bal api.MarketBalance) storagemarket.Balance { - return storagemarket.Balance{ +func toSharedBalance(bal api.MarketBalance) legacytypes.Balance { + return legacytypes.Balance{ Locked: bal.Locked, Available: big.Sub(bal.Escrow, bal.Locked), } diff --git a/go.mod b/go.mod index f42de4277..a573ad61d 100644 --- a/go.mod +++ b/go.mod @@ -2,33 +2,38 @@ module github.com/filecoin-project/boost go 1.20 +replace github.com/filecoin-project/filecoin-ffi => ./extern/filecoin-ffi + +replace github.com/filecoin-project/boostd-data => ./extern/boostd-data + +// replace github.com/filecoin-project/boost-graphsync => ../boost-graphsync + +// replace github.com/filecoin-project/boost-gfm => ../boost-gfm + require ( contrib.go.opencensus.io/exporter/prometheus v0.4.2 github.com/BurntSushi/toml v1.3.2 github.com/alecthomas/jsonschema v0.0.0-20200530073317-71f438968921 github.com/benbjohnson/clock v1.3.5 - github.com/buger/goterm v1.0.3 + github.com/buger/goterm v1.0.3 // indirect github.com/chzyer/readline v1.5.1 github.com/davecgh/go-spew v1.1.1 github.com/docker/go-units v0.5.0 github.com/dustin/go-humanize v1.0.1 github.com/etclabscore/go-openrpc-reflect v0.0.36 - github.com/fatih/color v1.13.0 github.com/filecoin-project/dagstore v0.7.0 github.com/filecoin-project/go-address v1.1.0 github.com/filecoin-project/go-bitfield v0.2.4 github.com/filecoin-project/go-cbor-util v0.0.1 github.com/filecoin-project/go-commp-utils v0.1.4 - github.com/filecoin-project/go-data-transfer v1.15.4-boost github.com/filecoin-project/go-fil-commcid v0.1.0 github.com/filecoin-project/go-fil-commp-hashhash v0.2.0 - github.com/filecoin-project/go-fil-markets v1.28.3 github.com/filecoin-project/go-jsonrpc v0.3.1 github.com/filecoin-project/go-padreader v0.0.1 github.com/filecoin-project/go-paramfetch v0.0.4 github.com/filecoin-project/go-state-types v0.12.8 github.com/filecoin-project/go-statestore v0.2.0 - github.com/filecoin-project/specs-actors v0.9.15 // indirect + github.com/filecoin-project/specs-actors v0.9.15 github.com/gbrlsnchs/jwt/v3 v3.0.1 github.com/golang/mock v1.6.0 github.com/google/uuid v1.3.0 @@ -36,7 +41,6 @@ require ( github.com/graph-gophers/graphql-go v1.3.0 github.com/graph-gophers/graphql-transport-ws v0.0.2 github.com/hashicorp/go-multierror v1.1.1 - github.com/hnlq715/golang-lru v0.4.0 github.com/ipfs/go-block-format v0.2.0 github.com/ipfs/go-blockservice v0.5.1 // indirect github.com/ipfs/go-cid v0.4.1 @@ -47,15 +51,12 @@ require ( github.com/ipfs/go-ipfs-chunker v0.0.5 // indirect github.com/ipfs/go-ipfs-ds-help v1.1.0 // indirect github.com/ipfs/go-ipfs-exchange-interface v0.2.0 // indirect - github.com/ipfs/go-ipfs-exchange-offline v0.3.0 // indirect - github.com/ipfs/go-ipfs-files v0.3.0 // indirect github.com/ipfs/go-ipld-format v0.6.0 github.com/ipfs/go-ipld-legacy v0.2.1 github.com/ipfs/go-libipfs v0.7.0 github.com/ipfs/go-log/v2 v2.5.1 github.com/ipfs/go-merkledag v0.11.0 // indirect github.com/ipfs/go-metrics-interface v0.0.1 - github.com/ipfs/go-unixfs v0.4.5 // indirect github.com/ipld/go-car v0.6.1 github.com/ipld/go-car/v2 v2.13.1 github.com/ipld/go-ipld-prime v0.21.0 @@ -77,7 +78,7 @@ require ( github.com/multiformats/go-multibase v0.2.0 github.com/multiformats/go-multicodec v0.9.0 github.com/multiformats/go-multihash v0.2.3 - github.com/multiformats/go-varint v0.0.7 // indirect + github.com/multiformats/go-varint v0.0.7 github.com/open-rpc/meta-schema v0.0.0-20201029221707-1b72ef2ea333 github.com/pressly/goose/v3 v3.14.0 github.com/prometheus/client_golang v1.16.0 @@ -119,7 +120,7 @@ require ( github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137 // indirect github.com/armon/go-metrics v0.3.9 // indirect github.com/beorn7/perks v1.0.1 // indirect - github.com/bep/debounce v1.2.1 // indirect + github.com/bep/debounce v1.2.1 github.com/boltdb/bolt v1.3.1 // indirect github.com/cespare/xxhash v1.1.0 // indirect github.com/cespare/xxhash/v2 v2.2.0 // indirect @@ -206,7 +207,6 @@ require ( github.com/ipfs/go-fs-lock v0.0.7 // indirect github.com/ipfs/go-ipfs-cmds v0.10.0 // indirect github.com/ipfs/go-ipfs-delay v0.0.1 // indirect - github.com/ipfs/go-ipfs-posinfo v0.0.1 // indirect github.com/ipfs/go-ipfs-pq v0.0.3 // indirect github.com/ipfs/go-ipfs-util v0.0.3 // indirect github.com/ipfs/go-ipld-cbor v0.1.0 @@ -247,7 +247,7 @@ require ( github.com/mailru/easyjson v0.7.7 // indirect github.com/marten-seemann/tcp v0.0.0-20210406111302-dfbc87cc63fd // indirect github.com/mattn/go-colorable v0.1.13 // indirect - github.com/mattn/go-isatty v0.0.19 // indirect + github.com/mattn/go-isatty v0.0.20 // indirect github.com/mattn/go-runewidth v0.0.14 // indirect github.com/matttproud/golang_protobuf_extensions v1.0.4 // indirect github.com/miekg/dns v1.1.55 // indirect @@ -266,7 +266,7 @@ require ( github.com/opencontainers/runtime-spec v1.1.0 // indirect github.com/opentracing/opentracing-go v1.2.0 // indirect github.com/pbnjay/memory v0.0.0-20210728143218-7b4eea64cf58 // indirect - github.com/petar/GoLLRB v0.0.0-20210522233825-ae3b015fd3e9 // indirect + github.com/petar/GoLLRB v0.0.0-20210522233825-ae3b015fd3e9 github.com/pkg/errors v0.9.1 // indirect github.com/pmezard/go-difflib v1.0.0 // indirect github.com/polydawn/refmt v0.89.0 // indirect @@ -274,7 +274,7 @@ require ( github.com/prometheus/common v0.44.0 // indirect github.com/prometheus/procfs v0.11.1 // indirect github.com/prometheus/statsd_exporter v0.22.7 // indirect - github.com/rivo/uniseg v0.2.0 // indirect + github.com/rivo/uniseg v0.4.4 // indirect github.com/rs/cors v1.7.0 github.com/russross/blackfriday/v2 v2.1.0 // indirect github.com/shirou/gopsutil v3.21.4-0.20210419000835-c7a38de76ee5+incompatible // indirect @@ -288,7 +288,7 @@ require ( github.com/valyala/bytebufferpool v1.0.0 // indirect github.com/valyala/fasttemplate v1.2.1 // indirect github.com/whyrusleeping/bencher v0.0.0-20190829221104-bb6607aa8bba // indirect - github.com/whyrusleeping/cbor v0.0.0-20171005072247-63513f603b11 // indirect + github.com/whyrusleeping/cbor v0.0.0-20171005072247-63513f603b11 github.com/whyrusleeping/chunker v0.0.0-20181014151217-fe64bd25879f // indirect github.com/whyrusleeping/go-keyspace v0.0.0-20160322163242-5b898ac5add1 // indirect github.com/whyrusleeping/ledger-filecoin-go v0.9.1-0.20201010031517-c3dcc1bddce4 // indirect @@ -300,7 +300,7 @@ require ( go.uber.org/zap v1.25.0 go4.org v0.0.0-20230225012048-214862532bf5 // indirect golang.org/x/mod v0.12.0 // indirect - golang.org/x/net v0.17.0 // indirect + golang.org/x/net v0.17.0 golang.org/x/sys v0.15.0 // indirect golang.org/x/term v0.15.0 golang.org/x/time v0.3.0 // indirect @@ -318,11 +318,9 @@ require ( ) require ( - github.com/filecoin-project/boost-gfm v1.26.7 github.com/filecoin-project/boost-graphsync v0.13.9 github.com/filecoin-project/boost/extern/boostd-data v0.0.0-20231124125934-3233c510357f github.com/filecoin-project/go-data-segment v0.0.1 - github.com/filecoin-project/go-data-transfer/v2 v2.0.0-rc7 github.com/filecoin-project/lotus v1.25.0 github.com/ipfs/boxo v0.12.0 github.com/ipfs/go-ds-leveldb v0.5.0 @@ -334,14 +332,15 @@ require ( github.com/ipni/go-libipni v0.5.2 github.com/ipni/ipni-cli v0.1.1 github.com/ipni/storetheindex v0.8.1 - github.com/schollz/progressbar/v3 v3.13.1 + github.com/schollz/progressbar/v3 v3.14.1 ) -require github.com/mitchellh/colorstring v0.0.0-20190213212951-d06e56a500db // indirect - require ( github.com/Jorropo/jsync v1.0.1 // indirect github.com/alexbrainman/goissue34681 v0.0.0-20191006012335-3fc7a47baff5 // indirect + github.com/fatih/color v1.13.0 // indirect + github.com/filecoin-project/go-data-transfer/v2 v2.0.0-rc7 // indirect + github.com/filecoin-project/go-fil-markets v1.28.3 // indirect github.com/filecoin-project/kubo-api-client v0.0.2-0.20230829103503-14448166d14d // indirect github.com/gammazero/channelqueue v0.2.1 // indirect github.com/gammazero/deque v0.2.1 // indirect @@ -362,6 +361,7 @@ require ( github.com/jackc/pgtype v1.10.0 // indirect github.com/jackc/puddle v1.2.1 // indirect github.com/lib/pq v1.10.9 // indirect + github.com/mitchellh/colorstring v0.0.0-20190213212951-d06e56a500db // indirect github.com/montanaflynn/stats v0.7.0 // indirect github.com/onsi/ginkgo/v2 v2.11.0 // indirect github.com/puzpuzpuz/xsync/v2 v2.4.0 // indirect diff --git a/go.sum b/go.sum index 87a0e4578..2b4fe63d2 100644 --- a/go.sum +++ b/go.sum @@ -309,17 +309,12 @@ github.com/fatih/color v1.8.0/go.mod h1:3l45GVGkyrnYNl9HoIjnp2NnNWvh6hLAqD8yTfGj github.com/fatih/color v1.13.0 h1:8LOYc1KYPPmyKMuN8QV2DNRWNbLo6LZ0iLs8+mlH53w= github.com/fatih/color v1.13.0/go.mod h1:kLAiJbzzSOZDVNGyDpeOxJ47H46qBXwg5ILebYFFOfk= github.com/felixge/httpsnoop v1.0.3 h1:s/nj+GCswXYzN5v2DpNMuMQYe+0DDwt5WVCU6CWBdXk= -github.com/filecoin-project/boost-gfm v1.26.7 h1:ENJEqx1OzY072QnUP37YrGVmUiCewRwHAjbtTxyW74Y= -github.com/filecoin-project/boost-gfm v1.26.7/go.mod h1:OhG2y7WeDx3KU9DPjgWllS+3/ospPjm8/XDrvN6uOfk= github.com/filecoin-project/boost-graphsync v0.13.9 h1:RQepfTlffLGUmp3Ff7VosYrWUKPLiz++GGV2D/gIfuw= github.com/filecoin-project/boost-graphsync v0.13.9/go.mod h1:bc2M5ZLZJtXHl8kjnqtn4L1MsdEqpJErDaIeY0bJ9wk= github.com/filecoin-project/boost/extern/boostd-data v0.0.0-20231124125934-3233c510357f h1:8dd0yAadyeOL5Qd42XhEwD60UKvIFkY2MLhef/IaeOk= github.com/filecoin-project/boost/extern/boostd-data v0.0.0-20231124125934-3233c510357f/go.mod h1:MyzvfYWAH0OAyf95TLUWYq3cO3vm/TVzDS57GKQi47o= github.com/filecoin-project/dagstore v0.7.0 h1:IS0R+69za8dguYWeqz/MI+nb7ONpk03tAkxPCBXEKm0= github.com/filecoin-project/dagstore v0.7.0/go.mod h1:YKn4qXih+/2xQWpfJsaKGOi4POw5vH5grDmfPCCnx8g= -github.com/filecoin-project/filecoin-ffi v0.30.4-0.20200910194244-f640612a1a1f/go.mod h1:+If3s2VxyjZn+KGGZIoRXBDSFQ9xL404JBJGf4WhEj0= -github.com/filecoin-project/filecoin-ffi v1.25.0-dev.1 h1:5e3Ic9K3P784zbEQgVdZ7od1h9EErpH357Ee7LMhlrQ= -github.com/filecoin-project/filecoin-ffi v1.25.0-dev.1/go.mod h1:pr0xMh3m/uvi65h+F6GObPsS9t1nDvNZkBwbRdoXaco= github.com/filecoin-project/go-address v0.0.3/go.mod h1:jr8JxKsYx+lQlQZmF5i2U0Z+cGQ59wMIps/8YW/lDj8= github.com/filecoin-project/go-address v0.0.5/go.mod h1:jr8JxKsYx+lQlQZmF5i2U0Z+cGQ59wMIps/8YW/lDj8= github.com/filecoin-project/go-address v1.1.0 h1:ofdtUtEsNxkIxkDw67ecSmvtzaVSdcea4boAmLbnHfE= @@ -349,13 +344,10 @@ github.com/filecoin-project/go-crypto v0.0.1 h1:AcvpSGGCgjaY8y1az6AMfKQWreF/pWO2 github.com/filecoin-project/go-crypto v0.0.1/go.mod h1:+viYnvGtUTgJRdy6oaeF4MTFKAfatX071MPDPBL11EQ= github.com/filecoin-project/go-data-segment v0.0.1 h1:1wmDxOG4ubWQm3ZC1XI5nCon5qgSq7Ra3Rb6Dbu10Gs= github.com/filecoin-project/go-data-segment v0.0.1/go.mod h1:H0/NKbsRxmRFBcLibmABv+yFNHdmtl5AyplYLnb0Zv4= -github.com/filecoin-project/go-data-transfer v1.15.4-boost h1:rGsPDeDk0nbzLOPn/9iCIrhLNy69Vkr9tRBcetM4kd0= -github.com/filecoin-project/go-data-transfer v1.15.4-boost/go.mod h1:S5Es9uoD+3TveYyGjxZInAF6mSQtRjNzezV7Y7Sh8X0= github.com/filecoin-project/go-data-transfer/v2 v2.0.0-rc7 h1:v+zJS5B6pA3ptWZS4t8tbt1Hz9qENnN4nVr1w99aSWc= github.com/filecoin-project/go-data-transfer/v2 v2.0.0-rc7/go.mod h1:V3Y4KbttaCwyg1gwkP7iai8CbQx4mZUGjd3h9GZWLKE= github.com/filecoin-project/go-ds-versioning v0.1.2 h1:to4pTadv3IeV1wvgbCbN6Vqd+fu+7tveXgv/rCEZy6w= github.com/filecoin-project/go-ds-versioning v0.1.2/go.mod h1:C9/l9PnB1+mwPa26BBVpCjG/XQCB0yj/q5CK2J8X1I4= -github.com/filecoin-project/go-fil-commcid v0.0.0-20200716160307-8f644712406f/go.mod h1:Eaox7Hvus1JgPrL5+M3+h7aSPHc0cVqpSxA+TxIEpZQ= github.com/filecoin-project/go-fil-commcid v0.0.0-20201016201715-d41df56b4f6a/go.mod h1:Eaox7Hvus1JgPrL5+M3+h7aSPHc0cVqpSxA+TxIEpZQ= github.com/filecoin-project/go-fil-commcid v0.1.0 h1:3R4ds1A9r6cr8mvZBfMYxTS88OqLYEo6roi+GiIeOh8= github.com/filecoin-project/go-fil-commcid v0.1.0/go.mod h1:Eaox7Hvus1JgPrL5+M3+h7aSPHc0cVqpSxA+TxIEpZQ= @@ -379,12 +371,12 @@ github.com/filecoin-project/go-paramfetch v0.0.4 h1:H+Me8EL8T5+79z/KHYQQcT8NVOzY github.com/filecoin-project/go-paramfetch v0.0.4/go.mod h1:1FH85P8U+DUEmWk1Jkw3Bw7FrwTVUNHk/95PSPG+dts= github.com/filecoin-project/go-retrieval-types v1.2.0 h1:fz6DauLVP3GRg7UuW7HZ6sE+GTmaUW70DTXBF1r9cK0= github.com/filecoin-project/go-state-types v0.0.0-20200903145444-247639ffa6ad/go.mod h1:IQ0MBPnonv35CJHtWSN3YY1Hz2gkPru1Q9qoaYLxx9I= -github.com/filecoin-project/go-state-types v0.0.0-20200904021452-1883f36ca2f4/go.mod h1:IQ0MBPnonv35CJHtWSN3YY1Hz2gkPru1Q9qoaYLxx9I= github.com/filecoin-project/go-state-types v0.0.0-20200928172055-2df22083d8ab/go.mod h1:ezYnPf0bNkTsDibL/psSz5dy4B5awOJ/E7P2Saeep8g= github.com/filecoin-project/go-state-types v0.0.0-20201102161440-c8033295a1fc/go.mod h1:ezYnPf0bNkTsDibL/psSz5dy4B5awOJ/E7P2Saeep8g= github.com/filecoin-project/go-state-types v0.1.0/go.mod h1:ezYnPf0bNkTsDibL/psSz5dy4B5awOJ/E7P2Saeep8g= github.com/filecoin-project/go-state-types v0.1.6/go.mod h1:UwGVoMsULoCK+bWjEdd/xLCvLAQFBC7EDT477SKml+Q= github.com/filecoin-project/go-state-types v0.1.10/go.mod h1:UwGVoMsULoCK+bWjEdd/xLCvLAQFBC7EDT477SKml+Q= +github.com/filecoin-project/go-state-types v0.11.2-0.20230712101859-8f37624fa540/go.mod h1:SyNPwTsU7I22gL2r0OAPcImvLoTVfgRwdK/Y5rR1zz8= github.com/filecoin-project/go-state-types v0.12.8 h1:W/UObdAsv+LbB9EfyLg92DSYoatzUWmlfV8FGyh30VA= github.com/filecoin-project/go-state-types v0.12.8/go.mod h1:gR2NV0CSGSQwopxF+3In9nDh1sqvoYukLcs5vK0AHCA= github.com/filecoin-project/go-statemachine v0.0.0-20200925024713-05bd7c71fbfe/go.mod h1:FGwQgZAt2Gh5mjlwJUlVB62JeYdo+if0xWxSEfBD9ig= @@ -401,7 +393,6 @@ github.com/filecoin-project/lotus v1.25.0 h1:okHySAq5Z2ZCRtooIBmqi5jujFBoXzuay+K github.com/filecoin-project/lotus v1.25.0/go.mod h1:7How7BIQEbV/2gHJBjsNQ7ynHb7zXUmtO7QYImo4L3w= github.com/filecoin-project/pubsub v1.0.0 h1:ZTmT27U07e54qV1mMiQo4HDr0buo8I1LDHBYLXlsNXM= github.com/filecoin-project/pubsub v1.0.0/go.mod h1:GkpB33CcUtUNrLPhJgfdy4FDx4OMNR9k+46DHx/Lqrg= -github.com/filecoin-project/specs-actors v0.9.4/go.mod h1:BStZQzx5x7TmCkLv0Bpa07U6cPKol6fd3w9KjMPZ6Z4= github.com/filecoin-project/specs-actors v0.9.13/go.mod h1:TS1AW/7LbG+615j4NsjMK1qlpAwaFsG9w0V2tg2gSao= github.com/filecoin-project/specs-actors v0.9.15-0.20220514164640-94e0d5e123bd/go.mod h1:pjGEe3QlWtK20ju/aFRsiArbMX6Cn8rqEhhsiCM9xYE= github.com/filecoin-project/specs-actors v0.9.15 h1:3VpKP5/KaDUHQKAMOg4s35g/syDaEBueKLws0vbsjMc= @@ -694,8 +685,6 @@ github.com/hashicorp/raft v1.3.10/go.mod h1:J8naEwc6XaaCfts7+28whSeRvCqTd6e20BlC github.com/hashicorp/raft-boltdb v0.0.0-20171010151810-6e5ba93211ea h1:xykPFhrBAS2J0VBzVa5e80b5ZtYuNQtgXjN40qBZlD4= github.com/hashicorp/raft-boltdb v0.0.0-20171010151810-6e5ba93211ea/go.mod h1:pNv7Wc3ycL6F5oOWn+tPGo2gWD4a5X+yp/ntwdKLjRk= github.com/hashicorp/serf v0.8.2/go.mod h1:6hOLApaqBFA1NXqRQAsxw9QxuDEvNxSQRwA/JwenrHc= -github.com/hnlq715/golang-lru v0.4.0 h1:gyo/wIvLE6Upf1wucAfwTjpR+BQ5Lli2766H2MnNPv0= -github.com/hnlq715/golang-lru v0.4.0/go.mod h1:RBkgDAtlu0SgTPvpb4VW2/RQnkCBMRD3Lr6B9RhsAS8= github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= github.com/hudl/fargo v1.3.0/go.mod h1:y3CKSmjA+wD2gak7sUSXTAoopbhU08POFhmITJgmKTg= github.com/huin/goupnp v1.0.0/go.mod h1:n9v9KO1tAxYH82qOn+UTIFQDmx5n1Zxd/ClZDMX7Bnc= @@ -792,7 +781,6 @@ github.com/ipfs/go-fs-lock v0.0.7 h1:6BR3dajORFrFTkb5EpCUFIAypsoxpGpDSVUdFwzgL9U github.com/ipfs/go-fs-lock v0.0.7/go.mod h1:Js8ka+FNYmgQRLrRXzU3CB/+Csr1BwrRilEcvYrHhhc= github.com/ipfs/go-graphsync v0.15.1 h1:7v4VfRQ/8pKzPuE0wHeMaWhKu8D/RlezIrzvGWIBtHQ= github.com/ipfs/go-graphsync v0.15.1/go.mod h1:eUIYS0OKkdBbG4vHhfGkY3lZ7h1G5Dlwd+HxTCe18vA= -github.com/ipfs/go-hamt-ipld v0.1.1/go.mod h1:1EZCr2v0jlCnhpa+aZ0JZYp8Tt2w16+JJOAVz17YcDk= github.com/ipfs/go-ipfs-blockstore v0.0.1/go.mod h1:d3WClOmRQKFnJ0Jz/jj/zmksX0ma1gROTlovZKBmN08= github.com/ipfs/go-ipfs-blockstore v0.1.0/go.mod h1:5aD0AvHPi7mZc6Ci1WCAhiBQu2IsfTduLl+422H6Rqw= github.com/ipfs/go-ipfs-blockstore v0.2.1/go.mod h1:jGesd8EtCM3/zPgx+qr0/feTXGUeRai6adgwC+Q+JvE= @@ -826,7 +814,6 @@ github.com/ipfs/go-ipfs-exchange-offline v0.3.0/go.mod h1:MOdJ9DChbb5u37M1IcbrRB github.com/ipfs/go-ipfs-files v0.0.3/go.mod h1:INEFm0LL2LWXBhNJ2PMIIb2w45hpXgPjNoE7yA8Y1d4= github.com/ipfs/go-ipfs-files v0.0.4/go.mod h1:INEFm0LL2LWXBhNJ2PMIIb2w45hpXgPjNoE7yA8Y1d4= github.com/ipfs/go-ipfs-files v0.3.0 h1:fallckyc5PYjuMEitPNrjRfpwl7YFt69heCOUhsbGxQ= -github.com/ipfs/go-ipfs-files v0.3.0/go.mod h1:xAUtYMwB+iu/dtf6+muHNSFQCJG2dSiStR2P6sn9tIM= github.com/ipfs/go-ipfs-posinfo v0.0.1 h1:Esoxj+1JgSjX0+ylc0hUmJCOv6V2vFoZiETLR6OtpRs= github.com/ipfs/go-ipfs-posinfo v0.0.1/go.mod h1:SwyeVP+jCwiDu0C313l/8jg6ZxM0qqtlt2a0vILTc1A= github.com/ipfs/go-ipfs-pq v0.0.1/go.mod h1:LWIqQpqfRG3fNc5XsnIhz/wQ2XXGyugQwls7BgUmUfY= @@ -853,6 +840,7 @@ github.com/ipfs/go-ipld-format v0.0.1/go.mod h1:kyJtbkDALmFHv3QR6et67i35QzO3S0dC github.com/ipfs/go-ipld-format v0.0.2/go.mod h1:4B6+FM2u9OJ9zCV+kSbgFAZlOrv1Hqbf0INGQgiKf9k= github.com/ipfs/go-ipld-format v0.2.0/go.mod h1:3l3C1uKoadTPbeNfrDi+xMInYKlx2Cvg1BuydPSdzQs= github.com/ipfs/go-ipld-format v0.3.0/go.mod h1:co/SdBE8h99968X0hViiw1MNlh6fvxxnHpvVLnH7jSM= +github.com/ipfs/go-ipld-format v0.4.0/go.mod h1:co/SdBE8h99968X0hViiw1MNlh6fvxxnHpvVLnH7jSM= github.com/ipfs/go-ipld-format v0.6.0 h1:VEJlA2kQ3LqFSIm5Vu6eIlSxD/Ze90xtc4Meten1F5U= github.com/ipfs/go-ipld-format v0.6.0/go.mod h1:g4QVMTn3marU3qXchwjpKPKgJv+zF+OlaKMyhJ4LHPg= github.com/ipfs/go-ipld-legacy v0.1.0/go.mod h1:86f5P/srAmh9GcIcWQR9lfFLZPrIyyXQeVlOWeeWEuI= @@ -898,7 +886,6 @@ github.com/ipfs/go-peertaskqueue v0.8.1/go.mod h1:Oxxd3eaK279FxeydSPPVGHzbwVeHja github.com/ipfs/go-unixfs v0.2.2-0.20190827150610-868af2e9e5cb/go.mod h1:IwAAgul1UQIcNZzKPYZWOCijryFBeCV79cNubPzol+k= github.com/ipfs/go-unixfs v0.3.1/go.mod h1:h4qfQYzghiIc8ZNFKiLMFWOTzrWIAtzYQ59W/pCFf1o= github.com/ipfs/go-unixfs v0.4.5 h1:wj8JhxvV1G6CD7swACwSKYa+NgtdWC1RUit+gFnymDU= -github.com/ipfs/go-unixfs v0.4.5/go.mod h1:BIznJNvt/gEx/ooRMI4Us9K8+qeGO7vx1ohnbk8gjFg= github.com/ipfs/go-unixfsnode v1.4.0/go.mod h1:qc7YFFZ8tABc58p62HnIYbUMwj9chhUuFWmxSokfePo= github.com/ipfs/go-unixfsnode v1.9.0 h1:ubEhQhr22sPAKO2DNsyVBW7YB/zA8Zkif25aBvz8rc8= github.com/ipfs/go-unixfsnode v1.9.0/go.mod h1:HxRu9HYHOjK6HUqFBAi++7DVoWAHn0o4v/nZ/VA+0g8= @@ -1368,9 +1355,8 @@ github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Ky github.com/mattn/go-isatty v0.0.13/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= -github.com/mattn/go-isatty v0.0.17/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= -github.com/mattn/go-isatty v0.0.19 h1:JITubQf0MOLdlGRuRq+jtsDlekdYPia9ZFsB8h/APPA= -github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= +github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY= +github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= github.com/mattn/go-pointer v0.0.1/go.mod h1:2zXcozF6qYGgmsG+SeTZz3oAbFLdD3OWqnUbNvJZAlc= github.com/mattn/go-runewidth v0.0.2/go.mod h1:LwmH8dsx7+W8Uxz3IHJYH5QSwggIsqBzpuz5H//U1FU= github.com/mattn/go-runewidth v0.0.7/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m2gUSrubnMI= @@ -1669,11 +1655,11 @@ github.com/raulk/clock v1.1.0/go.mod h1:3MpVxdZ/ODBQDxbN+kzshf5OSZwPjtMDx6BBXBmO github.com/raulk/go-watchdog v1.3.0 h1:oUmdlHxdkXRJlwfG0O9omj8ukerm8MEQavSiDTEtBsk= github.com/raulk/go-watchdog v1.3.0/go.mod h1:fIvOnLbF0b0ZwkB9YU4mOW9Did//4vPZtDqv66NfsMU= github.com/rcrowley/go-metrics v0.0.0-20181016184325-3113b8401b8a/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= -github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE= github.com/rivo/uniseg v0.1.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= -github.com/rivo/uniseg v0.2.0 h1:S1pD9weZBuJdFmowNwbpi7BJ8TNftyUImj/0WQi72jY= github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= +github.com/rivo/uniseg v0.4.4 h1:8TfxU8dW6PdqD27gjM8MVNuicgxIjxpm4K7x4jp8sis= +github.com/rivo/uniseg v0.4.4/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88= github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg= github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc= @@ -1692,8 +1678,8 @@ github.com/rwcarlsen/goexif v0.0.0-20190401172101-9e8deecbddbd/go.mod h1:hPqNNc0 github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= github.com/samuel/go-zookeeper v0.0.0-20190923202752-2cc03de413da/go.mod h1:gi+0XIa01GRL2eRQVjQkKGqKF3SF9vZR/HnPullcV2E= github.com/satori/go.uuid v1.2.0/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0= -github.com/schollz/progressbar/v3 v3.13.1 h1:o8rySDYiQ59Mwzy2FELeHY5ZARXZTVJC7iHD6PEFUiE= -github.com/schollz/progressbar/v3 v3.13.1/go.mod h1:xvrbki8kfT1fzWzBT/UZd9L6GA+jdL7HAgq2RFnO6fQ= +github.com/schollz/progressbar/v3 v3.14.1 h1:VD+MJPCr4s3wdhTc7OEJ/Z3dAeBzJ7yKH/P4lC5yRTI= +github.com/schollz/progressbar/v3 v3.14.1/go.mod h1:Zc9xXneTzWXF81TGoqL71u0sBPjULtEHYtj/WVgVy8E= github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc= github.com/sercand/kuberesolver v2.4.0+incompatible h1:WE2OlRf6wjLxHwNkkFLQGaZcVLEXjMjBPjjEU5vksH8= github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAmXWZgo= @@ -1798,7 +1784,6 @@ github.com/tidwall/match v1.0.1/go.mod h1:LujAq0jyVjBy028G1WhWfIzbpQfMO8bBZ6Tyb0 github.com/tidwall/match v1.1.1 h1:+Ho715JplO36QYgwN9PGYNhgZvoUSc9X2c80KVTi+GA= github.com/tidwall/pretty v1.0.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk= github.com/tidwall/pretty v1.2.0 h1:RWIZEg2iJ8/g6fDDYzMpobmaoGh5OLl4AXtGUGPcqCs= -github.com/tj/go-spin v1.1.0/go.mod h1:Mg1mzmePZm4dva8Qz60H2lHwmJ2loum4VIrLgVnKwh4= github.com/tklauser/go-sysconf v0.3.5 h1:uu3Xl4nkLzQfXNsWn15rPc/HQCJKObbt1dKJeWp3vU4= github.com/tklauser/go-sysconf v0.3.5/go.mod h1:MkWzOF4RMCshBAMXuhXJs64Rte09mITnppBXY/rYEFI= github.com/tklauser/numcpus v0.2.2 h1:oyhllyrScuYI6g+h/zUvNXNp1wy7x8qQy3t/piefldA= @@ -1878,8 +1863,6 @@ github.com/whyrusleeping/multiaddr-filter v0.0.0-20160516205228-e903e4adabd7 h1: github.com/whyrusleeping/multiaddr-filter v0.0.0-20160516205228-e903e4adabd7/go.mod h1:X2c0RVCI1eSUFI8eLcY3c0423ykwiUdxLJtkDvruhjI= github.com/x-cray/logrus-prefixed-formatter v0.5.2/go.mod h1:2duySbKsL6M18s5GU7VPsoEPHyzalCE06qoARUCeBBE= github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU= -github.com/xlab/c-for-go v0.0.0-20200718154222-87b0065af829/go.mod h1:h/1PEBwj7Ym/8kOuMWvO2ujZ6Lt+TMbySEXNhjjR87I= -github.com/xlab/pkgconfig v0.0.0-20170226114623-cea12a0fd245/go.mod h1:C+diUUz7pxhNY6KAoLgrTYARGWnt82zWTylZlxT92vk= github.com/xorcare/golden v0.6.0/go.mod h1:7T39/ZMvaSEZlBPoYfVFmsBLmUl3uz9IuzWj/U6FtvQ= github.com/xorcare/golden v0.6.1-0.20191112154924-b87f686d7542 h1:oWgZJmC1DorFZDpfMfWg7xk29yEOZiXmo/wZl+utTI8= github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q= @@ -2273,6 +2256,7 @@ golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.14.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= golang.org/x/sys v0.15.0 h1:h48lPFYpsTvQJZF4EKyI4aLHaev3CxivZmv7yZig9pc= golang.org/x/sys v0.15.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= golang.org/x/term v0.0.0-20201117132131-f5c789dd3221/go.mod h1:Nr5EML6q2oocZ2LXRh80K7BxOlk5/8JxuGnuhpl+muw= @@ -2281,7 +2265,7 @@ golang.org/x/term v0.0.0-20201210144234-2321bbc49cbf/go.mod h1:bj7SfCRtBDWHUb9sn golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.1.0/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= -golang.org/x/term v0.6.0/go.mod h1:m6U89DPEgQRMq3DNkDClhWw02AUbt2daBVO4cn4Hv9U= +golang.org/x/term v0.14.0/go.mod h1:TySc+nGkYR6qt8km8wUhuFRTVSMIX3XPR58y2lC8vww= golang.org/x/term v0.15.0 h1:y/Oo/a/q3IXu26lQgl04j/gjuBDOBlx7X6Om1j2CPW4= golang.org/x/term v0.15.0/go.mod h1:BDl952bC7+uMoWR75FIrCDx79TPU9oHkTZ9yRbYOrX0= golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= @@ -2355,7 +2339,6 @@ golang.org/x/tools v0.0.0-20200512131952-2bc93b1c0c88/go.mod h1:EkVYQZoAsY45+roY golang.org/x/tools v0.0.0-20200515010526-7d3b6ebf133d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200618134242-20370b0cb4b2/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= -golang.org/x/tools v0.0.0-20200711155855-7342f9734a7d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= golang.org/x/tools v0.0.0-20200729194436-6467de6f59a7/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= golang.org/x/tools v0.0.0-20200825202427-b303f430e36d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= @@ -2519,6 +2502,7 @@ gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.0-20200615113413-eeeca48fe776/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.0/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gotest.tools v2.2.0+incompatible h1:VsBPFP1AI068pPrMxtb/S8Zkgf9xEmTLJjfM+P5UIEo= @@ -2541,20 +2525,15 @@ lukechampine.com/blake3 v1.2.1 h1:YuqqRuaqsGV71BV/nm9xlI0MKUv4QC54jQnBChWbGnI= lukechampine.com/blake3 v1.2.1/go.mod h1:0OFRp7fBtAylGVCO40o87sbupkyIGgbpv1+M1k1LM6k= lukechampine.com/uint128 v1.3.0 h1:cDdUVfRwDUDovz610ABgFD17nXD4/uDgVHl2sC3+sbo= modernc.org/cc v1.0.0 h1:nPibNuDEx6tvYrUAtvDTTw98rx5juGsa5zuDnKwEEQQ= -modernc.org/cc v1.0.0/go.mod h1:1Sk4//wdnYJiUIxnW8ddKpaOJCF37yAdqYnkxUpaYxw= modernc.org/cc/v3 v3.41.0 h1:QoR1Sn3YWlmA1T4vLaKZfawdVtSiGx8H+cEojbC7v1Q= modernc.org/ccgo/v3 v3.16.14 h1:af6KNtFgsVmnDYrWk3PQCS9XT6BXe7o3ZFJKkIKvXNQ= -modernc.org/golex v1.0.0/go.mod h1:b/QX9oBD/LhixY6NDh+IdGv17hgB+51fET1i2kPSmvk= modernc.org/libc v1.24.1 h1:uvJSeCKL/AgzBo2yYIPPTy82v21KgGnizcGYfBHaNuM= -modernc.org/mathutil v1.1.1/go.mod h1:mZW8CKdRPY1v87qxC/wUdX5O1qDzXMP5TH3wjfpga6E= modernc.org/mathutil v1.6.0 h1:fRe9+AmYlaej+64JsEEhoWuAYBkOtQiMEU7n/XgfYi4= modernc.org/memory v1.6.0 h1:i6mzavxrE9a30whzMfwf7XWVODx2r5OYXvU46cirX7o= modernc.org/opt v0.1.3 h1:3XOZf2yznlhC+ibLltsDGzABUGVx8J6pnFMS3E4dcq4= modernc.org/sqlite v1.24.0 h1:EsClRIWHGhLTCX44p+Ri/JLD+vFGo0QGjasg2/F9TlI= -modernc.org/strutil v1.1.0/go.mod h1:lstksw84oURvj9y3tn8lGvRxyRC1S2+g5uuIzNfIOBs= modernc.org/strutil v1.1.3 h1:fNMm+oJklMGYfU9Ylcywl0CO5O6nTfaowNsh2wpPjzY= modernc.org/token v1.1.0 h1:Xl7Ap9dKaEs5kLoOQeQmPWevfnk/DM5qcLcYlA8ys6Y= -modernc.org/xc v1.0.0/go.mod h1:mRNCo0bvLjGhHO9WsyuKVU4q0ceiDDDoEeWDJHrNx8I= rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8= rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0= rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA= diff --git a/go.work.sum b/go.work.sum index 4400982cd..58ab4db7e 100644 --- a/go.work.sum +++ b/go.work.sum @@ -649,6 +649,7 @@ github.com/filecoin-project/boost/extern/boostd-data v0.0.0-20231009154452-ca8da github.com/filecoin-project/filecoin-ffi v0.30.4-0.20220519234331-bfd1f5f9fe38/go.mod h1:GM5pXRYvQM7wyH6V2WtPnJ2k1jt+qotRkWLxBSRCOuE= github.com/filecoin-project/go-dagaggregator-unixfs v0.3.0 h1:UXLtBUnPa61LkNa2GqhP+aJ53bOnHP/dzg6/wk2rnsA= github.com/filecoin-project/go-dagaggregator-unixfs v0.3.0/go.mod h1:UTWmEgyqq7RMx56AeHY/uEoLq1dJTPAirjyBPas4IQQ= +github.com/filecoin-project/go-data-transfer v1.15.2 h1:PzqsFr2Q/onMGKrGh7TtRT0dKsJcVJrioJJnjnKmxlk= github.com/filecoin-project/go-indexer-core v0.2.16 h1:1SmJVhfHTsi0CC+U6JdyjIIQtOqmKvCl/tqpI3gI+18= github.com/filecoin-project/go-legs v0.4.9 h1:9ccbv5zDPqMviEpSpf0TdfKKI64TMYGSiuY2A1EXHFY= github.com/filecoin-project/go-retrieval-types v1.2.0/go.mod h1:ojW6wSw2GPyoRDBGqw1K6JxUcbfa5NOSIiyQEeh7KK0= @@ -881,6 +882,7 @@ github.com/ipfs/go-ipfs v0.12.1 h1:stT4AJCiaTS2a+yL382g1IB8Gm+jJFqe7Ssvf+L9KNw= github.com/ipfs/go-ipfs-cmds v0.9.0/go.mod h1:SBFHK8WNwC416QWH9Vz1Ql42SSMAOqKpaHUMBu3jpLo= github.com/ipfs/go-ipfs-config v0.18.0 h1:Ta1aNGNEq6RIvzbw7dqzCVZJKb7j+Dd35JFnAOCpT8g= github.com/ipfs/go-ipfs-files v0.2.0/go.mod h1:vT7uaQfIsprKktzbTPLnIsd+NGw9ZbYwSq0g3N74u0M= +github.com/ipfs/go-ipfs-files v0.3.0/go.mod h1:xAUtYMwB+iu/dtf6+muHNSFQCJG2dSiStR2P6sn9tIM= github.com/ipfs/go-ipfs-http-client v0.4.0 h1:LNuVbFoKfCohCmcNImml3byM3PpTxTT7RPrv/UoDFkI= github.com/ipfs/go-ipfs-keystore v0.1.0 h1:gfuQUO/cyGZgZIHE6OrJas4OnwuxXCqJG7tI0lrB5Qc= github.com/ipfs/go-ipfs-keystore v0.1.0/go.mod h1:LvLw7Qhnb0RlMOfCzK6OmyWxICip6lQ06CCmdbee75U= @@ -904,6 +906,7 @@ github.com/ipfs/go-path v0.3.1/go.mod h1:eNLsxJEEMxn/CDzUJ6wuNl+6No6tEUhOZcPKsZs github.com/ipfs/go-pinning-service-http-client v0.1.0 h1:Au0P4NglL5JfzhNSZHlZ1qra+IcJyO3RWMd9EYCwqSY= github.com/ipfs/go-todocounter v0.0.1 h1:kITWA5ZcQZfrUnDNkRn04Xzh0YFaDFXsoO2A81Eb6Lw= github.com/ipfs/go-unixfs v0.4.4/go.mod h1:TSG7G1UuT+l4pNj91raXAPkX0BhJi3jST1FDTfQ5QyM= +github.com/ipfs/go-unixfs v0.4.5/go.mod h1:BIznJNvt/gEx/ooRMI4Us9K8+qeGO7vx1ohnbk8gjFg= github.com/ipfs/go-unixfsnode v1.5.1/go.mod h1:ed79DaG9IEuZITJVQn4U6MZDftv6I3ygUBLPfhEbHvk= github.com/ipfs/go-unixfsnode v1.7.1/go.mod h1:PVfoyZkX1B34qzT3vJO4nsLUpRCyhnMuHBznRcXirlk= github.com/ipfs/go-unixfsnode v1.7.4/go.mod h1:PVfoyZkX1B34qzT3vJO4nsLUpRCyhnMuHBznRcXirlk= @@ -1090,6 +1093,7 @@ github.com/marten-seemann/qtls-go1-19 v0.1.0 h1:rLFKD/9mp/uq1SYGYuVZhm83wkmU95pK github.com/marten-seemann/webtransport-go v0.1.1 h1:TnyKp3pEXcDooTaNn4s9dYpMJ7kMnTp7k5h+SgYP/mc= github.com/marten-seemann/webtransport-go v0.1.1/go.mod h1:kBEh5+RSvOA4troP1vyOVBWK4MIMzDICXVrvCPrYcrM= github.com/matryer/is v1.4.0 h1:sosSmIWwkYITGrxZ25ULNDeKiMNzFSr4V/eqBQP0PeE= +github.com/mattn/go-isatty v0.0.17/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= github.com/mattn/go-tty v0.0.0-20180907095812-13ff1204f104 h1:d8RFOZ2IiFtFWBcKEHAFYJcPTf0wY5q0exFNJZVWa1U= github.com/mattn/go-xmlrpc v0.0.3 h1:Y6WEMLEsqs3RviBrAa1/7qmbGB7DVD3brZIbqMbQdGY= github.com/mdlayher/genetlink v1.0.0 h1:OoHN1OdyEIkScEmRgxLEe2M9U8ClMytqA5niynLtfj0= @@ -1232,6 +1236,7 @@ github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475 h1:N/ElC8H3+5X github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= github.com/retailnext/hllpp v1.0.1-0.20180308014038-101a6d2f8b52 h1:RnWNS9Hlm8BIkjr6wx8li5abe0fr73jljLycdfemTp0= +github.com/rivo/uniseg v0.2.0 h1:S1pD9weZBuJdFmowNwbpi7BJ8TNftyUImj/0WQi72jY= github.com/rogpeppe/fastuuid v1.2.0 h1:Ppwyp6VYCF1nvBTXL3trRso7mXMlRrw9ooo375wvi2s= github.com/rogpeppe/go-internal v1.1.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs= @@ -1250,6 +1255,8 @@ github.com/samber/lo v1.36.0 h1:4LaOxH1mHnbDGhTVE0i1z8v/lWaQW8AIfOD3HU4mSaw= github.com/samber/lo v1.36.0/go.mod h1:HLeWcJRRyLKp3+/XBJvOrerCQn9mhdKMHyd7IRlgeQ8= github.com/samuel/go-zookeeper v0.0.0-20190923202752-2cc03de413da h1:p3Vo3i64TCLY7gIfzeQaUJ+kppEO5WQG3cL8iE8tGHU= github.com/satori/go.uuid v1.2.0 h1:0uYX9dsZ2yD7q2RtLRtPSdGDWzjeM3TbMJP9utgA0ww= +github.com/schollz/progressbar/v3 v3.13.1 h1:o8rySDYiQ59Mwzy2FELeHY5ZARXZTVJC7iHD6PEFUiE= +github.com/schollz/progressbar/v3 v3.13.1/go.mod h1:xvrbki8kfT1fzWzBT/UZd9L6GA+jdL7HAgq2RFnO6fQ= github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529 h1:nn5Wsu0esKSJiIVhscUtVbo7ada43DJhG55ua/hjS5I= github.com/seccomp/libseccomp-golang v0.9.1 h1:NJjM5DNFOs0s3kYE1WUOr6G8V97sdt46rlXTMfXGWBo= github.com/segmentio/asm v1.2.0 h1:9BQrFxC+YOHJlTlHGkTrFWf59nbL3XnCoFLTwDCI7ys= @@ -1532,32 +1539,39 @@ golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.9.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.11.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.13.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/term v0.4.0/go.mod h1:9P2UbLfCdcvo3p/nzKvsmas4TnlujnuoV9hGgYzW1lQ= -golang.org/x/term v0.11.0/go.mod h1:zC9APTIj3jG3FdV/Ons+XE1riIZXG4aZ4GTHiPZJPIU= +golang.org/x/sys v0.14.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= +golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= golang.org/x/term v0.13.0/go.mod h1:LTmsnFJwVN6bCy1rVCoS+qHT1HhALEFxKncY3WNNh4U= +golang.org/x/term v0.14.0/go.mod h1:TySc+nGkYR6qt8km8wUhuFRTVSMIX3XPR58y2lC8vww= +golang.org/x/term v0.15.0 h1:y/Oo/a/q3IXu26lQgl04j/gjuBDOBlx7X6Om1j2CPW4= +golang.org/x/term v0.15.0/go.mod h1:BDl952bC7+uMoWR75FIrCDx79TPU9oHkTZ9yRbYOrX0= golang.org/x/text v0.5.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= +golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= golang.org/x/text v0.8.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= golang.org/x/text v0.10.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= golang.org/x/text v0.12.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= golang.org/x/text v0.13.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= -golang.org/x/time v0.0.0-20220722155302-e5dcc9cfc0b9/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= -golang.org/x/tools v0.0.0-20190614205625-5aca471b1d59/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= +golang.org/x/time v0.3.0 h1:rg5rLMjNzMS1RkNLzCG38eapWhnYLFYXDXj2gOlr8j4= +golang.org/x/time v0.3.0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/tools v0.0.0-20191112195655-aa38f8e97acc/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20200904185747-39188db58858/go.mod h1:Cj7w3i3Rnn0Xh82ur9kSqwfTHTeVxaDqrfMjpcNT6bE= golang.org/x/tools v0.0.0-20201110124207-079ba7bd75cd/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= -golang.org/x/tools v0.0.0-20201112185108-eeaa07dd7696/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20201201161351-ac6f37ff4c2a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20201208233053-a543418bbed2/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20210105154028-b0ab187a4818/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20210112230658-8b4aab62c064/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.2.0/go.mod h1:y4OqIKeOV/fWJetJ8bXPU1sEVniLMIyDAZWeHdV+NTA= golang.org/x/tools v0.3.0/go.mod h1:/rWhSS2+zyEVwoJf8YAX6L2f0ntZ7Kn/mGgAWcipA5k= golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU= golang.org/x/tools v0.7.0/go.mod h1:4pg6aUX35JBAogB10C9AtvVL+qowtN4pT3CGSQex14s= golang.org/x/tools v0.9.1/go.mod h1:owI94Op576fPu3cIGQeHs3joujW/2Oc6MtlxbF5dfNc= golang.org/x/tools v0.9.3/go.mod h1:owI94Op576fPu3cIGQeHs3joujW/2Oc6MtlxbF5dfNc= +golang.org/x/tools v0.12.1-0.20230815132531-74c255bcf846/go.mod h1:Sc0INKfu04TlqNoRA1hgpFZbhYXHPr4V5DzpSBTPqQM= gonum.org/v1/gonum v0.9.3/go.mod h1:TZumC3NeyVQskjXqmyWt4S3bINhy7B4eYwW69EbyX+0= gonum.org/v1/gonum v0.11.0/go.mod h1:fSG4YDCxxUZQJ7rKsQrj0gMOg00Il0Z96/qMA4bVQhA= +gonum.org/v1/gonum v0.13.0 h1:a0T3bh+7fhRyqeNbiC3qVHYmkiQgit3wnNan/2c0HMM= +gonum.org/v1/gonum v0.13.0/go.mod h1:/WPYRckkfWrhWefxyYTfrTtQR0KH4iyHNuzxqXAKyAU= gonum.org/v1/netlib v0.0.0-20190313105609-8cb42192e0e0 h1:OE9mWmgKkjJyEmDAAtGMPjXu+YNeGvK9VTSHY6+Qihc= gonum.org/v1/plot v0.9.0/go.mod h1:3Pcqqmp6RHvJI72kgb8fThyUnav364FOsdDo2aGW5lY= gonum.org/v1/plot v0.10.1 h1:dnifSs43YJuNMDzB7v8wV64O4ABBHReuAVAoBxqBqS4= @@ -1568,9 +1582,6 @@ google.golang.org/api v0.40.0/go.mod h1:fYKFpnQN0DsDSKRVRcQSDQNtqWPfM9i+zNPxepjR google.golang.org/api v0.41.0/go.mod h1:RkxM5lITDfTzmyKFPt+wGrCJbVfniCr2ool8kTBzRTU= google.golang.org/api v0.43.0/go.mod h1:nQsDGjRXMo4lvh5hP0TKqF244gqhGcr/YSIykhUk/94= google.golang.org/api v0.44.0/go.mod h1:EBOGZqzyhtvMDoxwS97ctnh0zUmYY6CxqXsc1AvkYD8= -google.golang.org/api v0.108.0/go.mod h1:2Ts0XTHNVWxypznxWOYUeI4g3WdP9Pk2Qk58+a/O9MY= -google.golang.org/api v0.110.0 h1:l+rh0KYUooe9JGbGVx71tbFo4SMbMTXK3I3ia2QSEeU= -google.golang.org/api v0.110.0/go.mod h1:7FC4Vvx1Mooxh8C5HWjzZHcavuS2f6pmJpZx60ca7iI= google.golang.org/api v0.114.0 h1:1xQPji6cO2E2vLiI+C/XiFAnsn1WV3mjaEwGLhi3grE= google.golang.org/api v0.114.0/go.mod h1:ifYI2ZsFK6/uGddGfAD5BMxlnkBqCmqHSDUVi45N5Yg= google.golang.org/appengine v1.6.7 h1:FZR1q0exgwxzPzp/aF+VccGrSfxfPpkBqjIIEq3ru6c= @@ -1587,24 +1598,24 @@ google.golang.org/genproto v0.0.0-20210319143718-93e7006c17a6/go.mod h1:FWY/as6D google.golang.org/genproto v0.0.0-20210402141018-6c239bbf2bb1/go.mod h1:9lPAdzaEmUacj36I+k7YKbEc5CXzPIeORRgDAUOu28A= google.golang.org/genproto v0.0.0-20220822174746-9e6da59bd2fc/go.mod h1:dbqgFATTzChvnt+ujMdZwITVAJHFtfyN1qUhDqEiIlk= google.golang.org/genproto v0.0.0-20230110181048-76db0878b65f/go.mod h1:RGgjbofJ8xD9Sq1VVhDM1Vok1vRONV+rg+CjzG4SZKM= -google.golang.org/genproto v0.0.0-20230124163310-31e0e69b6fc2/go.mod h1:RGgjbofJ8xD9Sq1VVhDM1Vok1vRONV+rg+CjzG4SZKM= -google.golang.org/genproto v0.0.0-20230209215440-0dfe4f8abfcc/go.mod h1:RGgjbofJ8xD9Sq1VVhDM1Vok1vRONV+rg+CjzG4SZKM= -google.golang.org/genproto v0.0.0-20230216225411-c8e22ba71e44/go.mod h1:8B0gmkoRebU8ukX6HP+4wrVQUY1+6PkQ44BSyIlflHA= google.golang.org/genproto v0.0.0-20230306155012-7f2fa6fef1f4/go.mod h1:NWraEVixdDnqcqQ30jipen1STv2r/n24Wb7twVTGR4s= google.golang.org/genproto v0.0.0-20230323212658-478b75c54725/go.mod h1:UUQDJDOlWu4KYeJZffbWgBkS1YFobzKbLVfK69pe0Ak= +google.golang.org/genproto v0.0.0-20230410155749-daa745c078e1 h1:KpwkzHKEF7B9Zxg18WzOa7djJ+Ha5DzthMyZYQfEn2A= +google.golang.org/genproto v0.0.0-20230410155749-daa745c078e1/go.mod h1:nKE/iIaLqn2bQwXBg8f1g2Ylh6r5MN5CmZvuzZCgsCU= google.golang.org/grpc v1.34.0/go.mod h1:WotjhfgOW/POjDeRt8vscBtXq+2VjORFy659qA51WJ8= google.golang.org/grpc v1.35.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= google.golang.org/grpc v1.36.1/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= -google.golang.org/grpc v1.37.0/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= google.golang.org/grpc v1.42.0/go.mod h1:k+4IHHFw41K8+bbowsex27ge2rCb65oeWqe4jJ590SU= google.golang.org/grpc v1.48.0/go.mod h1:vN9eftEi1UMyUsIF80+uQXhHjbXYbm0uXoFCACuMGWk= -google.golang.org/grpc v1.51.0/go.mod h1:wgNDFcnuBGmxLKI/qn4T+m5BtEBYXJPvibbUPsAIPww= google.golang.org/grpc v1.53.0/go.mod h1:OnIrk0ipVdj4N5d9IUoFUx72/VlD7+jUsHwZgwSMQpw= google.golang.org/grpc v1.54.0/go.mod h1:PUSEXI6iWghWaB6lXM4knEgpJNu2qUcKfDtNci3EC2g= google.golang.org/grpc v1.55.0/go.mod h1:iYEXKGkEBhg1PjZQvoYEVPTDkHo1/bjTnfwTeGONTY8= -google.golang.org/grpc/cmd/protoc-gen-go-grpc v1.1.0 h1:M1YKkFIboKNieVO5DLUEVzQfGwJD30Nv2jfUgzb5UcE= -google.golang.org/grpc/cmd/protoc-gen-go-grpc v1.1.0/go.mod h1:6Kw0yEErY5E/yWrBtf03jp27GLLJujG4z/JK95pnjjw= +google.golang.org/grpc v1.56.3 h1:8I4C0Yq1EjstUzUJzpcRVbuYA2mODtEmpWiQoN/b2nc= +google.golang.org/grpc v1.56.3/go.mod h1:I9bI3vqKfayGqPUAwGdOSu7kt6oIJLixfffKrpXqQ9s= +google.golang.org/protobuf v1.31.0 h1:g0LDEJHgrBl9N9r17Ru3sqWhkIx2NB67okBHPwC7hs8= +google.golang.org/protobuf v1.31.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= gopkg.in/alecthomas/kingpin.v2 v2.2.6 h1:jMFz6MfLP0/4fUyZle81rXUoxOBFi19VUFKVDOQfozc= +gopkg.in/cheggaaa/pb.v1 v1.0.28 h1:n1tBJnnK2r7g9OW2btFH91V92STTUevLXYFb8gy9EMk= gopkg.in/errgo.v2 v2.1.0 h1:0vLT13EuvQ0hNvakwLuFZ/jYrLp5F3kcWHXdRggjCE8= gopkg.in/fsnotify.v1 v1.4.7 h1:xOHLXZwVvI9hhs+cLKq5+I5onOuwQLhQwiu63xxlHs4= gopkg.in/gcfg.v1 v1.2.3 h1:m8OOJ4ccYHnx2f4gQwpno8nAX5OGOh7RLaaz0pj3Ogs= @@ -1619,10 +1630,15 @@ gopkg.in/src-d/go-cli.v0 v0.0.0-20181105080154-d492247bbc0d h1:mXa4inJUuWOoA4uER gopkg.in/src-d/go-log.v1 v1.0.1 h1:heWvX7J6qbGWbeFS/aRmiy1eYaT+QMV6wNvHDyMjQV4= gopkg.in/warnings.v0 v0.1.2 h1:wFXVbFY8DY5/xOe1ECiWdKCzZlxgshcYVNkBHstARME= gopkg.in/yaml.v1 v1.0.0-20140924161607-9f9df34309c0 h1:POO/ycCATvegFmVuPpQzZFJ+pGZeX22Ufu6fibxDVjU= +gopkg.in/yaml.v1 v1.0.0-20140924161607-9f9df34309c0/go.mod h1:WDnlLJ4WF5VGsH/HVa3CI79GS0ol3YnhVnKP89i0kNg= +gopkg.in/yaml.v3 v3.0.0-20200615113413-eeeca48fe776/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gotest.tools v2.2.0+incompatible h1:VsBPFP1AI068pPrMxtb/S8Zkgf9xEmTLJjfM+P5UIEo= gotest.tools/v3 v3.4.0 h1:ZazjZUfuVeZGLAmlKKuyv3IKP5orXcwtOwDQH6YVr6o= gotest.tools/v3 v3.4.0/go.mod h1:CtbdzLSsqVhDgMtKsx03ird5YTGB3ar27v0u/yKBW5g= grpc.go4.org v0.0.0-20170609214715-11d0a25b4919 h1:tmXTu+dfa+d9Evp8NpJdgOy6+rt8/x4yG7qPBrtNfLY= honnef.co/go/tools v0.1.3 h1:qTakTkI6ni6LFD5sBwwsdSO+AQqbSIxOauHTTQKZ/7o= +howett.net/plist v1.0.0 h1:7CrbWYbPPO/PyNy38b2EB/+gYbjCe2DXBxgtOOZbSQM= +howett.net/plist v1.0.0/go.mod h1:lqaXoTrLY4hg8tnEzNru53gicrbv7rrk+2xJA/7hw9g= lukechampine.com/uint128 v1.3.0/go.mod h1:c4eWIwlEGaxC/+H1VguhU4PHXNWDCDMUlWdIWl2j1gk= modernc.org/cc/v3 v3.41.0/go.mod h1:Ni4zjJYJ04CDOhG7dn640WGfwBzfE0ecX8TyMB0Fv0Y= modernc.org/ccgo/v3 v3.16.14/go.mod h1:mPDSujUIaTNWQSG4eqKw+atqLOEbma6Ncsa94WbC9zo= @@ -1633,7 +1649,6 @@ modernc.org/httpfs v1.0.6 h1:AAgIpFZRXuYnkjftxTAZwMIiwEqAfk8aVB2/oA6nAeM= modernc.org/lex v1.0.0 h1:w0dxp18i1q+aSE7GkepvwzvVWTLoCIQ2oDgTFAV2JZU= modernc.org/lexer v1.0.0 h1:D2xE6YTaH7aiEC7o/+rbx6qTAEr1uY83peKwkamIdQ0= modernc.org/libc v1.24.1/go.mod h1:FmfO1RLrU3MHJfyi9eYYmZBfi/R+tqZ6+hQ3yQQUkak= -modernc.org/mathutil v1.5.0/go.mod h1:mZW8CKdRPY1v87qxC/wUdX5O1qDzXMP5TH3wjfpga6E= modernc.org/mathutil v1.6.0/go.mod h1:Ui5Q9q1TR2gFm0AQRqQUaBWFLAhQpCwNcuhBOSedWPo= modernc.org/memory v1.6.0/go.mod h1:PkUhL0Mugw21sHPeskwZW4D6VscE/GQJOnIpCnW6pSU= modernc.org/opt v0.1.3/go.mod h1:WdSiB5evDcignE70guQKxYUl14mgWtbClRi5wmkkTX0= diff --git a/gql/module.go b/gql/module.go new file mode 100644 index 000000000..efe888cf7 --- /dev/null +++ b/gql/module.go @@ -0,0 +1,68 @@ +package gql + +import ( + "context" + + "github.com/filecoin-project/boost/cmd/lib" + "github.com/filecoin-project/boost/db" + "github.com/filecoin-project/boost/fundmanager" + "github.com/filecoin-project/boost/indexprovider" + "github.com/filecoin-project/boost/lib/legacy" + "github.com/filecoin-project/boost/lib/mpoolmonitor" + "github.com/filecoin-project/boost/markets/storageadapter" + "github.com/filecoin-project/boost/node/config" + "github.com/filecoin-project/boost/piecedirectory" + "github.com/filecoin-project/boost/retrievalmarket/rtvllog" + "github.com/filecoin-project/boost/sectorstatemgr" + "github.com/filecoin-project/boost/storagemanager" + "github.com/filecoin-project/boost/storagemarket" + "github.com/filecoin-project/boost/storagemarket/sealingpipeline" + "github.com/filecoin-project/boost/storagemarket/storedask" + "github.com/filecoin-project/lotus/api/v1api" + "github.com/filecoin-project/lotus/node/repo" + blocks "github.com/ipfs/go-block-format" + "github.com/ipfs/go-cid" + provider "github.com/ipni/index-provider" + "github.com/libp2p/go-libp2p/core/host" + "go.uber.org/fx" +) + +func NewGraphqlServer(cfg *config.Boost) func(lc fx.Lifecycle, r repo.LockedRepo, h host.Host, prov *storagemarket.Provider, ddProv *storagemarket.DirectDealsProvider, dealsDB *db.DealsDB, directDealsDB *db.DirectDealsDB, logsDB *db.LogsDB, retDB *rtvllog.RetrievalLogDB, plDB *db.ProposalLogsDB, fundsDB *db.FundsDB, fundMgr *fundmanager.FundManager, storageMgr *storagemanager.StorageManager, publisher *storageadapter.DealPublisher, spApi sealingpipeline.API, legacyDeals legacy.LegacyDealManager, piecedirectory *piecedirectory.PieceDirectory, indexProv provider.Interface, idxProvWrapper *indexprovider.Wrapper, fullNode v1api.FullNode, bg BlockGetter, ssm *sectorstatemgr.SectorStateMgr, mpool *mpoolmonitor.MpoolMonitor, mma *lib.MultiMinerAccessor, sask storedask.StoredAsk) *Server { + return func(lc fx.Lifecycle, r repo.LockedRepo, h host.Host, prov *storagemarket.Provider, ddProv *storagemarket.DirectDealsProvider, dealsDB *db.DealsDB, directDealsDB *db.DirectDealsDB, logsDB *db.LogsDB, retDB *rtvllog.RetrievalLogDB, plDB *db.ProposalLogsDB, fundsDB *db.FundsDB, fundMgr *fundmanager.FundManager, + storageMgr *storagemanager.StorageManager, publisher *storageadapter.DealPublisher, spApi sealingpipeline.API, + legacyDeals legacy.LegacyDealManager, piecedirectory *piecedirectory.PieceDirectory, + indexProv provider.Interface, idxProvWrapper *indexprovider.Wrapper, fullNode v1api.FullNode, bg BlockGetter, + ssm *sectorstatemgr.SectorStateMgr, mpool *mpoolmonitor.MpoolMonitor, mma *lib.MultiMinerAccessor, sask storedask.StoredAsk) *Server { + + resolverCtx, cancel := context.WithCancel(context.Background()) + resolver := NewResolver(resolverCtx, cfg, r, h, dealsDB, directDealsDB, logsDB, retDB, plDB, fundsDB, fundMgr, storageMgr, spApi, prov, ddProv, legacyDeals, piecedirectory, publisher, indexProv, idxProvWrapper, fullNode, ssm, mpool, mma, sask) + svr := NewServer(cfg, resolver, bg) + + lc.Append(fx.Hook{ + OnStart: svr.Start, + OnStop: func(ctx context.Context) error { + cancel() + return svr.Stop(ctx) + }, + }) + + return svr + } +} + +func NewBlockGetter(pd *piecedirectory.PieceDirectory) BlockGetter { + return &pdBlockGetter{pd: pd} +} + +type pdBlockGetter struct { + pd *piecedirectory.PieceDirectory +} + +func (p *pdBlockGetter) Get(ctx context.Context, c cid.Cid) (blocks.Block, error) { + bz, err := p.pd.BlockstoreGet(ctx, c) + if err != nil { + return nil, err + } + + return blocks.NewBlockWithCid(bz, c) +} diff --git a/gql/resolver.go b/gql/resolver.go index 31e376428..cec532331 100644 --- a/gql/resolver.go +++ b/gql/resolver.go @@ -9,8 +9,6 @@ import ( "time" "github.com/dustin/go-humanize" - "github.com/filecoin-project/boost-gfm/piecestore" - gfm_storagemarket "github.com/filecoin-project/boost-gfm/storagemarket" "github.com/filecoin-project/boost/cmd/lib" "github.com/filecoin-project/boost/db" "github.com/filecoin-project/boost/fundmanager" @@ -20,13 +18,13 @@ import ( "github.com/filecoin-project/boost/lib/mpoolmonitor" "github.com/filecoin-project/boost/markets/storageadapter" "github.com/filecoin-project/boost/node/config" - "github.com/filecoin-project/boost/node/modules/dtypes" "github.com/filecoin-project/boost/piecedirectory" "github.com/filecoin-project/boost/retrievalmarket/rtvllog" "github.com/filecoin-project/boost/sectorstatemgr" "github.com/filecoin-project/boost/storagemanager" "github.com/filecoin-project/boost/storagemarket" "github.com/filecoin-project/boost/storagemarket/sealingpipeline" + "github.com/filecoin-project/boost/storagemarket/storedask" "github.com/filecoin-project/boost/storagemarket/types" "github.com/filecoin-project/boost/storagemarket/types/dealcheckpoints" "github.com/filecoin-project/boost/transport" @@ -65,11 +63,8 @@ type resolver struct { fundMgr *fundmanager.FundManager storageMgr *storagemanager.StorageManager provider *storagemarket.Provider + legacyDeals legacy.LegacyDealManager ddProvider *storagemarket.DirectDealsProvider - legacyDeals *legacy.LegacyDealsManager - legacyProv gfm_storagemarket.StorageProvider - legacyDT dtypes.ProviderDataTransfer - ps piecestore.PieceStore ssm *sectorstatemgr.SectorStateMgr piecedirectory *piecedirectory.PieceDirectory publisher *storageadapter.DealPublisher @@ -79,9 +74,10 @@ type resolver struct { fullNode v1api.FullNode mpool *mpoolmonitor.MpoolMonitor mma *lib.MultiMinerAccessor + askProv storedask.StoredAsk } -func NewResolver(ctx context.Context, cfg *config.Boost, r lotus_repo.LockedRepo, h host.Host, dealsDB *db.DealsDB, directDealsDB *db.DirectDealsDB, logsDB *db.LogsDB, retDB *rtvllog.RetrievalLogDB, plDB *db.ProposalLogsDB, fundsDB *db.FundsDB, fundMgr *fundmanager.FundManager, storageMgr *storagemanager.StorageManager, spApi sealingpipeline.API, provider *storagemarket.Provider, ddProvider *storagemarket.DirectDealsProvider, legacyDeals *legacy.LegacyDealsManager, legacyProv gfm_storagemarket.StorageProvider, legacyDT dtypes.ProviderDataTransfer, ps piecestore.PieceStore, piecedirectory *piecedirectory.PieceDirectory, publisher *storageadapter.DealPublisher, indexProv provider.Interface, idxProvWrapper *indexprovider.Wrapper, fullNode v1api.FullNode, ssm *sectorstatemgr.SectorStateMgr, mpool *mpoolmonitor.MpoolMonitor, mma *lib.MultiMinerAccessor) *resolver { +func NewResolver(ctx context.Context, cfg *config.Boost, r lotus_repo.LockedRepo, h host.Host, dealsDB *db.DealsDB, directDealsDB *db.DirectDealsDB, logsDB *db.LogsDB, retDB *rtvllog.RetrievalLogDB, plDB *db.ProposalLogsDB, fundsDB *db.FundsDB, fundMgr *fundmanager.FundManager, storageMgr *storagemanager.StorageManager, spApi sealingpipeline.API, provider *storagemarket.Provider, ddProvider *storagemarket.DirectDealsProvider, legacyDeals legacy.LegacyDealManager, piecedirectory *piecedirectory.PieceDirectory, publisher *storageadapter.DealPublisher, indexProv provider.Interface, idxProvWrapper *indexprovider.Wrapper, fullNode v1api.FullNode, ssm *sectorstatemgr.SectorStateMgr, mpool *mpoolmonitor.MpoolMonitor, mma *lib.MultiMinerAccessor, assk storedask.StoredAsk) *resolver { return &resolver{ ctx: ctx, cfg: cfg, @@ -98,9 +94,6 @@ func NewResolver(ctx context.Context, cfg *config.Boost, r lotus_repo.LockedRepo provider: provider, ddProvider: ddProvider, legacyDeals: legacyDeals, - legacyProv: legacyProv, - legacyDT: legacyDT, - ps: ps, piecedirectory: piecedirectory, publisher: publisher, spApi: spApi, @@ -110,6 +103,7 @@ func NewResolver(ctx context.Context, cfg *config.Boost, r lotus_repo.LockedRepo ssm: ssm, mpool: mpool, mma: mma, + askProv: assk, } } diff --git a/gql/resolver_ask.go b/gql/resolver_ask.go index 696929fa3..b68849391 100644 --- a/gql/resolver_ask.go +++ b/gql/resolver_ask.go @@ -5,8 +5,8 @@ import ( "fmt" "time" - "github.com/filecoin-project/boost-gfm/storagemarket" "github.com/filecoin-project/boost/gql/types" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/lotus/build" "github.com/graph-gophers/graphql-go" @@ -27,7 +27,7 @@ func (r *resolver) StorageAsk(ctx context.Context) (*storageAskResolver, error) return nil, fmt.Errorf("getting chain head: %w", err) } - signedAsk := r.legacyProv.GetAsk() + signedAsk := r.askProv.GetAsk(r.provider.Address) ask := signedAsk.Ask expTimeEpochs := ask.Expiry - head.Height() expTime := time.Now().Add(time.Duration(expTimeEpochs) * time.Duration(build.BlockDelaySecs) * time.Second) @@ -49,8 +49,8 @@ type storageAskUpdate struct { MaxPieceSize *types.Uint64 } -func (r *resolver) StorageAskUpdate(args struct{ Update storageAskUpdate }) (bool, error) { - signedAsk := r.legacyProv.GetAsk() +func (r *resolver) StorageAskUpdate(ctx context.Context, args struct{ Update storageAskUpdate }) (bool, error) { + signedAsk := r.askProv.GetAsk(r.provider.Address) ask := signedAsk.Ask dur := 87660 * time.Hour // 10 years @@ -58,7 +58,7 @@ func (r *resolver) StorageAskUpdate(args struct{ Update storageAskUpdate }) (boo price := ask.Price verifiedPrice := ask.VerifiedPrice - var opts []storagemarket.StorageAskOption + var opts []legacytypes.StorageAskOption update := args.Update if update.Price != nil { @@ -68,13 +68,13 @@ func (r *resolver) StorageAskUpdate(args struct{ Update storageAskUpdate }) (boo verifiedPrice = (*update.VerifiedPrice).Int } if update.MinPieceSize != nil { - opts = append(opts, storagemarket.MinPieceSize(abi.PaddedPieceSize(*update.MinPieceSize))) + opts = append(opts, legacytypes.MinPieceSize(abi.PaddedPieceSize(*update.MinPieceSize))) } if update.MaxPieceSize != nil { - opts = append(opts, storagemarket.MaxPieceSize(abi.PaddedPieceSize(*update.MaxPieceSize))) + opts = append(opts, legacytypes.MaxPieceSize(abi.PaddedPieceSize(*update.MaxPieceSize))) } - err := r.legacyProv.SetAsk(price, verifiedPrice, duration, opts...) + err := r.askProv.SetAsk(ctx, price, verifiedPrice, duration, r.provider.Address, opts...) if err != nil { return false, fmt.Errorf("setting ask: %w", err) } diff --git a/gql/resolver_dealpublish.go b/gql/resolver_dealpublish.go index e47fc4d33..e28cb4e85 100644 --- a/gql/resolver_dealpublish.go +++ b/gql/resolver_dealpublish.go @@ -104,7 +104,7 @@ func (r *resolver) DealPublish(ctx context.Context) (*dealPublishResolver, error // If there are any legacy deals to look up if len(legacyDealIDs) > 0 { // Get all deals from the legacy provider - legacyDeals, err := r.legacyProv.ListLocalDeals() + legacyDeals, err := r.legacyDeals.ListDeals() if err != nil { return nil, fmt.Errorf("getting legacy deals: %w", err) } @@ -158,7 +158,7 @@ func (r *resolver) DealPublish(ctx context.Context) (*dealPublishResolver, error Deals: basicDeals, Period: int32(pending.PublishPeriod.Seconds()), Start: graphql.Time{Time: pending.PublishPeriodStart}, - MaxDealsPerMsg: int32(r.cfg.LotusDealmaking.MaxDealsPerPublishMsg), + MaxDealsPerMsg: int32(r.cfg.Dealpublish.MaxDealsPerPublishMsg), }, nil } diff --git a/gql/resolver_legacy.go b/gql/resolver_legacy.go index b47001902..30186f8bd 100644 --- a/gql/resolver_legacy.go +++ b/gql/resolver_legacy.go @@ -5,14 +5,14 @@ import ( "fmt" "github.com/dustin/go-humanize" - "github.com/filecoin-project/boost-gfm/storagemarket" gqltypes "github.com/filecoin-project/boost/gql/types" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" "github.com/graph-gophers/graphql-go" "github.com/ipfs/go-cid" ) type legacyDealResolver struct { - storagemarket.MinerDeal + legacytypes.MinerDeal transferred uint64 } @@ -28,7 +28,7 @@ func (r *resolver) LegacyDeal(ctx context.Context, args struct{ ID graphql.ID }) return nil, fmt.Errorf("parsing deal signed proposal cid %s: %w", args.ID, err) } - dl, err := r.legacyProv.GetLocalDeal(signedPropCid) + dl, err := r.legacyDeals.ByPropCid(signedPropCid) if err != nil { return nil, fmt.Errorf("getting deal with signed proposal cid %s: %w", args.ID, err) } @@ -36,16 +36,8 @@ func (r *resolver) LegacyDeal(ctx context.Context, args struct{ ID graphql.ID }) return r.withTransferState(ctx, dl), nil } -func (r *resolver) withTransferState(ctx context.Context, dl storagemarket.MinerDeal) *legacyDealResolver { +func (r *resolver) withTransferState(ctx context.Context, dl legacytypes.MinerDeal) *legacyDealResolver { dr := &legacyDealResolver{MinerDeal: dl} - if dl.TransferChannelId != nil { - st, err := r.legacyDT.ChannelState(ctx, *dl.TransferChannelId) - if err != nil { - log.Warnw("getting transfer channel id %s: %s", *dl.TransferChannelId, err) - } else { - dr.transferred = st.Received() - } - } return dr } @@ -71,26 +63,26 @@ func (r *resolver) LegacyDeals(ctx context.Context, args dealsArgs) (*legacyDeal } // Get the total number of deals - dealCount, err := r.legacyProv.LocalDealCount() + dealCount, err := r.legacyDeals.DealCount(ctx) if err != nil { return nil, fmt.Errorf("getting deal count: %w", err) } var more bool - var pageDeals []storagemarket.MinerDeal + var pageDeals []legacytypes.MinerDeal if args.Query.Value != nil && *args.Query.Value != "" { // If there is a search query, assume the query is the deal // proposal cid and try to fetch the corresponding deal propCidQuery, err := cid.Parse(*args.Query.Value) if err == nil { - dl, err := r.legacyProv.GetLocalDeal(propCidQuery) + dl, err := r.legacyDeals.ByPropCid(propCidQuery) if err == nil { - pageDeals = []storagemarket.MinerDeal{dl} + pageDeals = []legacytypes.MinerDeal{dl} } } } else { // Get a page worth of deals, plus one extra so we can see if there are more deals - pageDeals, err = r.legacyProv.ListLocalDealsPage(startPropCid, offset, limit+1) + pageDeals, err = r.legacyDeals.ListLocalDealsPage(startPropCid, offset, limit+1) if err != nil { return nil, fmt.Errorf("getting page of deals: %w", err) } @@ -215,11 +207,11 @@ func (r *legacyDealResolver) InboundCARPath() string { } func (r *legacyDealResolver) Status() string { - return storagemarket.DealStates[r.State] + return legacytypes.DealStates[r.State] } func (r *legacyDealResolver) Message() string { - if r.MinerDeal.Message == "" && r.State == storagemarket.StorageDealTransferring { + if r.MinerDeal.Message == "" && r.State == legacytypes.StorageDealTransferring { switch r.transferred { case 0: return "Transferring" diff --git a/gql/resolver_legacy_storage.go b/gql/resolver_legacy_storage.go deleted file mode 100644 index 623decd15..000000000 --- a/gql/resolver_legacy_storage.go +++ /dev/null @@ -1,35 +0,0 @@ -package gql - -import ( - "errors" - "os" - "path" - - gqltypes "github.com/filecoin-project/boost/gql/types" - "github.com/filecoin-project/boost/util" - lotus_modules "github.com/filecoin-project/lotus/node/modules" -) - -type legacyStorageResolver struct { - Capacity gqltypes.Uint64 - Used gqltypes.Uint64 - MountPoint string -} - -// query: legacyStorage: [LegacyStorage] -func (r *resolver) LegacyStorage() (*legacyStorageResolver, error) { - stagingDir := path.Join(r.repo.Path(), lotus_modules.StagingAreaDirName) - used, err := util.DirSize(stagingDir) - if err != nil { - if !errors.Is(err, os.ErrNotExist) { - return nil, err - } - used = 0 - } - - return &legacyStorageResolver{ - Capacity: gqltypes.Uint64(r.cfg.LotusDealmaking.MaxStagingDealsBytes), - Used: gqltypes.Uint64(used), - MountPoint: stagingDir, - }, nil -} diff --git a/gql/resolver_rtvllog.go b/gql/resolver_rtvllog.go index 02eaa1a48..be062b29d 100644 --- a/gql/resolver_rtvllog.go +++ b/gql/resolver_rtvllog.go @@ -2,6 +2,7 @@ package gql import ( "context" + gqltypes "github.com/filecoin-project/boost/gql/types" "github.com/filecoin-project/boost/retrievalmarket/rtvllog" "github.com/graph-gophers/graphql-go" @@ -211,6 +212,6 @@ func (r *resolver) RetrievalLogsCount(ctx context.Context, args struct{ IsIndexe count, err := r.retDB.Count(ctx, isIndexer) return &retStateCount{ Count: int32(count), - Period: gqltypes.Uint64(r.cfg.Dealmaking.RetrievalLogDuration), + Period: gqltypes.Uint64(r.cfg.Retrievals.Graphsync.RetrievalLogDuration), }, err } diff --git a/gql/resolver_transfers.go b/gql/resolver_transfers.go index e7a20f306..ce1b3dd20 100644 --- a/gql/resolver_transfers.go +++ b/gql/resolver_transfers.go @@ -49,7 +49,7 @@ func (r *resolver) TransferStats(_ context.Context) *transferStats { }) } return &transferStats{ - HttpMaxConcurrentDownloads: int32(r.cfg.Dealmaking.HttpTransferMaxConcurrentDownloads), + HttpMaxConcurrentDownloads: int32(r.cfg.HttpDownload.HttpTransferMaxConcurrentDownloads), Stats: gqlStats, } } diff --git a/gql/schema.graphql b/gql/schema.graphql index e1d8d60d4..93e38da5f 100644 --- a/gql/schema.graphql +++ b/gql/schema.graphql @@ -305,12 +305,6 @@ type Storage { MountPoint: String! } -type LegacyStorage { - Capacity: Uint64! - Used: Uint64! - MountPoint: String! -} - type WaitDeal { ID: ID! Size: Uint64! @@ -597,9 +591,6 @@ type RootQuery { """Get storage space usage""" storage: Storage! - """Get storage space usage of deals made with legacy markets endpoint""" - legacyStorage: LegacyStorage! - """Get sealing pipeline state""" sealingpipeline: SealingPipeline! diff --git a/indexprovider/mock/mock.go b/indexprovider/mock/mock.go deleted file mode 100644 index 0e6e4e4cf..000000000 --- a/indexprovider/mock/mock.go +++ /dev/null @@ -1,273 +0,0 @@ -// Code generated by MockGen. DO NOT EDIT. -// Source: github.com/filecoin-project/boost-gfm/storagemarket (interfaces: StorageProvider) - -// Package mock is a generated GoMock package. -package mock - -import ( - context "context" - io "io" - reflect "reflect" - - shared "github.com/filecoin-project/boost-gfm/shared" - storagemarket "github.com/filecoin-project/boost-gfm/storagemarket" - abi "github.com/filecoin-project/go-state-types/abi" - big "github.com/filecoin-project/go-state-types/big" - gomock "github.com/golang/mock/gomock" - cid "github.com/ipfs/go-cid" -) - -// MockStorageProvider is a mock of StorageProvider interface. -type MockStorageProvider struct { - ctrl *gomock.Controller - recorder *MockStorageProviderMockRecorder -} - -// MockStorageProviderMockRecorder is the mock recorder for MockStorageProvider. -type MockStorageProviderMockRecorder struct { - mock *MockStorageProvider -} - -// NewMockStorageProvider creates a new mock instance. -func NewMockStorageProvider(ctrl *gomock.Controller) *MockStorageProvider { - mock := &MockStorageProvider{ctrl: ctrl} - mock.recorder = &MockStorageProviderMockRecorder{mock} - return mock -} - -// EXPECT returns an object that allows the caller to indicate expected use. -func (m *MockStorageProvider) EXPECT() *MockStorageProviderMockRecorder { - return m.recorder -} - -// AddStorageCollateral mocks base method. -func (m *MockStorageProvider) AddStorageCollateral(arg0 context.Context, arg1 big.Int) error { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "AddStorageCollateral", arg0, arg1) - ret0, _ := ret[0].(error) - return ret0 -} - -// AddStorageCollateral indicates an expected call of AddStorageCollateral. -func (mr *MockStorageProviderMockRecorder) AddStorageCollateral(arg0, arg1 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "AddStorageCollateral", reflect.TypeOf((*MockStorageProvider)(nil).AddStorageCollateral), arg0, arg1) -} - -// AnnounceAllDealsToIndexer mocks base method. -func (m *MockStorageProvider) AnnounceAllDealsToIndexer(arg0 context.Context) error { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "AnnounceAllDealsToIndexer", arg0) - ret0, _ := ret[0].(error) - return ret0 -} - -// AnnounceAllDealsToIndexer indicates an expected call of AnnounceAllDealsToIndexer. -func (mr *MockStorageProviderMockRecorder) AnnounceAllDealsToIndexer(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "AnnounceAllDealsToIndexer", reflect.TypeOf((*MockStorageProvider)(nil).AnnounceAllDealsToIndexer), arg0) -} - -// AnnounceDealToIndexer mocks base method. -func (m *MockStorageProvider) AnnounceDealToIndexer(arg0 context.Context, arg1 cid.Cid) error { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "AnnounceDealToIndexer", arg0, arg1) - ret0, _ := ret[0].(error) - return ret0 -} - -// AnnounceDealToIndexer indicates an expected call of AnnounceDealToIndexer. -func (mr *MockStorageProviderMockRecorder) AnnounceDealToIndexer(arg0, arg1 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "AnnounceDealToIndexer", reflect.TypeOf((*MockStorageProvider)(nil).AnnounceDealToIndexer), arg0, arg1) -} - -// GetAsk mocks base method. -func (m *MockStorageProvider) GetAsk() *storagemarket.SignedStorageAsk { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "GetAsk") - ret0, _ := ret[0].(*storagemarket.SignedStorageAsk) - return ret0 -} - -// GetAsk indicates an expected call of GetAsk. -func (mr *MockStorageProviderMockRecorder) GetAsk() *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetAsk", reflect.TypeOf((*MockStorageProvider)(nil).GetAsk)) -} - -// GetLocalDeal mocks base method. -func (m *MockStorageProvider) GetLocalDeal(arg0 cid.Cid) (storagemarket.MinerDeal, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "GetLocalDeal", arg0) - ret0, _ := ret[0].(storagemarket.MinerDeal) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// GetLocalDeal indicates an expected call of GetLocalDeal. -func (mr *MockStorageProviderMockRecorder) GetLocalDeal(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetLocalDeal", reflect.TypeOf((*MockStorageProvider)(nil).GetLocalDeal), arg0) -} - -// GetStorageCollateral mocks base method. -func (m *MockStorageProvider) GetStorageCollateral(arg0 context.Context) (storagemarket.Balance, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "GetStorageCollateral", arg0) - ret0, _ := ret[0].(storagemarket.Balance) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// GetStorageCollateral indicates an expected call of GetStorageCollateral. -func (mr *MockStorageProviderMockRecorder) GetStorageCollateral(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetStorageCollateral", reflect.TypeOf((*MockStorageProvider)(nil).GetStorageCollateral), arg0) -} - -// ImportDataForDeal mocks base method. -func (m *MockStorageProvider) ImportDataForDeal(arg0 context.Context, arg1 cid.Cid, arg2 io.Reader) error { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "ImportDataForDeal", arg0, arg1, arg2) - ret0, _ := ret[0].(error) - return ret0 -} - -// ImportDataForDeal indicates an expected call of ImportDataForDeal. -func (mr *MockStorageProviderMockRecorder) ImportDataForDeal(arg0, arg1, arg2 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "ImportDataForDeal", reflect.TypeOf((*MockStorageProvider)(nil).ImportDataForDeal), arg0, arg1, arg2) -} - -// ListLocalDeals mocks base method. -func (m *MockStorageProvider) ListLocalDeals() ([]storagemarket.MinerDeal, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "ListLocalDeals") - ret0, _ := ret[0].([]storagemarket.MinerDeal) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// ListLocalDeals indicates an expected call of ListLocalDeals. -func (mr *MockStorageProviderMockRecorder) ListLocalDeals() *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "ListLocalDeals", reflect.TypeOf((*MockStorageProvider)(nil).ListLocalDeals)) -} - -// ListLocalDealsPage mocks base method. -func (m *MockStorageProvider) ListLocalDealsPage(arg0 *cid.Cid, arg1, arg2 int) ([]storagemarket.MinerDeal, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "ListLocalDealsPage", arg0, arg1, arg2) - ret0, _ := ret[0].([]storagemarket.MinerDeal) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// ListLocalDealsPage indicates an expected call of ListLocalDealsPage. -func (mr *MockStorageProviderMockRecorder) ListLocalDealsPage(arg0, arg1, arg2 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "ListLocalDealsPage", reflect.TypeOf((*MockStorageProvider)(nil).ListLocalDealsPage), arg0, arg1, arg2) -} - -// LocalDealCount mocks base method. -func (m *MockStorageProvider) LocalDealCount() (int, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "LocalDealCount") - ret0, _ := ret[0].(int) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// LocalDealCount indicates an expected call of LocalDealCount. -func (mr *MockStorageProviderMockRecorder) LocalDealCount() *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "LocalDealCount", reflect.TypeOf((*MockStorageProvider)(nil).LocalDealCount)) -} - -// OnReady mocks base method. -func (m *MockStorageProvider) OnReady(arg0 shared.ReadyFunc) { - m.ctrl.T.Helper() - m.ctrl.Call(m, "OnReady", arg0) -} - -// OnReady indicates an expected call of OnReady. -func (mr *MockStorageProviderMockRecorder) OnReady(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "OnReady", reflect.TypeOf((*MockStorageProvider)(nil).OnReady), arg0) -} - -// RetryDealPublishing mocks base method. -func (m *MockStorageProvider) RetryDealPublishing(arg0 cid.Cid) error { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "RetryDealPublishing", arg0) - ret0, _ := ret[0].(error) - return ret0 -} - -// RetryDealPublishing indicates an expected call of RetryDealPublishing. -func (mr *MockStorageProviderMockRecorder) RetryDealPublishing(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "RetryDealPublishing", reflect.TypeOf((*MockStorageProvider)(nil).RetryDealPublishing), arg0) -} - -// SetAsk mocks base method. -func (m *MockStorageProvider) SetAsk(arg0, arg1 big.Int, arg2 abi.ChainEpoch, arg3 ...storagemarket.StorageAskOption) error { - m.ctrl.T.Helper() - varargs := []interface{}{arg0, arg1, arg2} - for _, a := range arg3 { - varargs = append(varargs, a) - } - ret := m.ctrl.Call(m, "SetAsk", varargs...) - ret0, _ := ret[0].(error) - return ret0 -} - -// SetAsk indicates an expected call of SetAsk. -func (mr *MockStorageProviderMockRecorder) SetAsk(arg0, arg1, arg2 interface{}, arg3 ...interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - varargs := append([]interface{}{arg0, arg1, arg2}, arg3...) - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "SetAsk", reflect.TypeOf((*MockStorageProvider)(nil).SetAsk), varargs...) -} - -// Start mocks base method. -func (m *MockStorageProvider) Start(arg0 context.Context) error { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "Start", arg0) - ret0, _ := ret[0].(error) - return ret0 -} - -// Start indicates an expected call of Start. -func (mr *MockStorageProviderMockRecorder) Start(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Start", reflect.TypeOf((*MockStorageProvider)(nil).Start), arg0) -} - -// Stop mocks base method. -func (m *MockStorageProvider) Stop() error { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "Stop") - ret0, _ := ret[0].(error) - return ret0 -} - -// Stop indicates an expected call of Stop. -func (mr *MockStorageProviderMockRecorder) Stop() *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Stop", reflect.TypeOf((*MockStorageProvider)(nil).Stop)) -} - -// SubscribeToEvents mocks base method. -func (m *MockStorageProvider) SubscribeToEvents(arg0 storagemarket.ProviderSubscriber) shared.Unsubscribe { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "SubscribeToEvents", arg0) - ret0, _ := ret[0].(shared.Unsubscribe) - return ret0 -} - -// SubscribeToEvents indicates an expected call of SubscribeToEvents. -func (mr *MockStorageProviderMockRecorder) SubscribeToEvents(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "SubscribeToEvents", reflect.TypeOf((*MockStorageProvider)(nil).SubscribeToEvents), arg0) -} diff --git a/indexprovider/wrapper.go b/indexprovider/wrapper.go index b94451e57..509660ba3 100644 --- a/indexprovider/wrapper.go +++ b/indexprovider/wrapper.go @@ -7,15 +7,16 @@ import ( "fmt" "net/url" "os" - "path/filepath" + "github.com/filecoin-project/boost/lib/legacy" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" + "github.com/filecoin-project/go-statemachine/fsm" "github.com/google/uuid" "go.uber.org/fx" "github.com/ipfs/go-datastore" "github.com/ipld/go-ipld-prime" - "github.com/filecoin-project/boost-gfm/storagemarket" "github.com/filecoin-project/boost/db" bdtypes "github.com/filecoin-project/boost/extern/boostd-data/svc/types" "github.com/filecoin-project/boost/markets/idxprov" @@ -43,15 +44,14 @@ import ( ) var log = logging.Logger("index-provider-wrapper") -var defaultDagStoreDir = "dagstore" type Wrapper struct { enabled bool cfg *config.Boost dealsDB *db.DealsDB + legacyProv legacy.LegacyDealManager directDealsDB *db.DirectDealsDB - legacyProv storagemarket.StorageProvider prov provider.Interface piecedirectory *piecedirectory.PieceDirectory ssm *sectorstatemgr.SectorStateMgr @@ -65,25 +65,21 @@ type Wrapper struct { } func NewWrapper(cfg *config.Boost) func(lc fx.Lifecycle, h host.Host, r repo.LockedRepo, dealsDB *db.DealsDB, - ssDB *db.SectorStateDB, legacyProv storagemarket.StorageProvider, prov provider.Interface, + ssDB *db.SectorStateDB, legacyProv legacy.LegacyDealManager, prov provider.Interface, piecedirectory *piecedirectory.PieceDirectory, ssm *sectorstatemgr.SectorStateMgr, meshCreator idxprov.MeshCreator, storageService lotus_modules.MinerStorageService) (*Wrapper, error) { return func(lc fx.Lifecycle, h host.Host, r repo.LockedRepo, dealsDB *db.DealsDB, - ssDB *db.SectorStateDB, legacyProv storagemarket.StorageProvider, prov provider.Interface, + ssDB *db.SectorStateDB, legacyProv legacy.LegacyDealManager, prov provider.Interface, piecedirectory *piecedirectory.PieceDirectory, ssm *sectorstatemgr.SectorStateMgr, meshCreator idxprov.MeshCreator, storageService lotus_modules.MinerStorageService) (*Wrapper, error) { - if cfg.DAGStore.RootDir == "" { - cfg.DAGStore.RootDir = filepath.Join(r.Path(), defaultDagStoreDir) - } - _, isDisabled := prov.(*DisabledIndexProvider) // bitswap is enabled if there is a bitswap peer id - bitswapEnabled := cfg.Dealmaking.BitswapPeerID != "" + bitswapEnabled := cfg.Retrievals.Bitswap.BitswapPeerID != "" // http is considered enabled if there is an http retrieval multiaddr set - httpEnabled := cfg.Dealmaking.HTTPRetrievalMultiaddr != "" + httpEnabled := cfg.Retrievals.HTTP.HTTPRetrievalMultiaddr != "" // setup bitswap extended provider if there is a public multi addr for bitswap w := &Wrapper{ @@ -208,7 +204,7 @@ func (w *Wrapper) handleUpdates(ctx context.Context, sectorUpdates map[abi.Secto } // Get deals by sector ID, whether they're legacy or boost deals -func (w *Wrapper) dealsBySectorID(ctx context.Context, legacyDeals map[abi.SectorID][]storagemarket.MinerDeal, sectorID abi.SectorID) ([]basicDealInfo, error) { +func (w *Wrapper) dealsBySectorID(ctx context.Context, legacyDeals map[abi.SectorID][]legacytypes.MinerDeal, sectorID abi.SectorID) ([]basicDealInfo, error) { // First query the boost database deals, err := w.dealsDB.BySectorID(ctx, sectorID) if err != nil { @@ -244,13 +240,13 @@ func (w *Wrapper) dealsBySectorID(ctx context.Context, legacyDeals map[abi.Secto // Iterate over all legacy deals and make a map of sector ID -> legacy deal. // To save memory, only include legacy deals with a sector ID that we know // we're going to query, ie the set of sector IDs in the stateUpdates map. -func (w *Wrapper) legacyDealsBySectorID(stateUpdates map[abi.SectorID]db.SealState) (map[abi.SectorID][]storagemarket.MinerDeal, error) { - legacyDeals, err := w.legacyProv.ListLocalDeals() +func (w *Wrapper) legacyDealsBySectorID(stateUpdates map[abi.SectorID]db.SealState) (map[abi.SectorID][]legacytypes.MinerDeal, error) { + legacyDeals, err := w.legacyProv.ListDeals() if err != nil { return nil, err } - bySectorID := make(map[abi.SectorID][]storagemarket.MinerDeal, len(legacyDeals)) + bySectorID := make(map[abi.SectorID][]legacytypes.MinerDeal, len(legacyDeals)) for _, deal := range legacyDeals { minerID, err := address.IDFromAddress(deal.Proposal.Provider) if err != nil { @@ -373,26 +369,26 @@ func (w *Wrapper) appendExtendedProviders(ctx context.Context, adBuilder *xprovi return err } var ep xproviders.Info - if len(w.cfg.Dealmaking.BitswapPublicAddresses) > 0 { - if w.cfg.Dealmaking.BitswapPrivKeyFile == "" { + if len(w.cfg.Retrievals.Bitswap.BitswapPublicAddresses) > 0 { + if w.cfg.Retrievals.Bitswap.BitswapPrivKeyFile == "" { return fmt.Errorf("missing required configuration key BitswapPrivKeyFile: " + "boost is configured with BitswapPublicAddresses but the BitswapPrivKeyFile configuration key is empty") } // we need the private key for bitswaps peerID in order to announce publicly - keyFile, err := os.ReadFile(w.cfg.Dealmaking.BitswapPrivKeyFile) + keyFile, err := os.ReadFile(w.cfg.Retrievals.Bitswap.BitswapPrivKeyFile) if err != nil { - return fmt.Errorf("opening BitswapPrivKeyFile %s: %w", w.cfg.Dealmaking.BitswapPrivKeyFile, err) + return fmt.Errorf("opening BitswapPrivKeyFile %s: %w", w.cfg.Retrievals.Bitswap.BitswapPrivKeyFile, err) } privKey, err := crypto.UnmarshalPrivateKey(keyFile) if err != nil { - return fmt.Errorf("unmarshalling BitswapPrivKeyFile %s: %w", w.cfg.Dealmaking.BitswapPrivKeyFile, err) + return fmt.Errorf("unmarshalling BitswapPrivKeyFile %s: %w", w.cfg.Retrievals.Bitswap.BitswapPrivKeyFile, err) } // setup an extended provider record, containing the booster-bitswap multi addr, // peer ID, private key for signing, and metadata ep = xproviders.Info{ - ID: w.cfg.Dealmaking.BitswapPeerID, - Addrs: w.cfg.Dealmaking.BitswapPublicAddresses, + ID: w.cfg.Retrievals.Bitswap.BitswapPeerID, + Addrs: w.cfg.Retrievals.Bitswap.BitswapPublicAddresses, Priv: privKey, Metadata: mbytes, } @@ -430,7 +426,7 @@ func (w *Wrapper) appendExtendedProviders(ctx context.Context, adBuilder *xprovi } var ep = xproviders.Info{ ID: w.h.ID().String(), - Addrs: []string{w.cfg.Dealmaking.HTTPRetrievalMultiaddr}, + Addrs: []string{w.cfg.Retrievals.HTTP.HTTPRetrievalMultiaddr}, Metadata: mbytes, Priv: key, } @@ -449,22 +445,57 @@ func (w *Wrapper) IndexerAnnounceAllDeals(ctx context.Context) error { } log.Info("announcing all legacy deals to Indexer") - err := w.legacyProv.AnnounceAllDealsToIndexer(ctx) - if err == nil { - log.Infof("finished announcing all legacy deals to Indexer") - } else { - log.Warnw("failed to announce legacy deals to Indexer", "err", err) + + legacyDeals, err := w.legacyProv.ListDeals() + if err != nil { + return fmt.Errorf("failed to get the list of legacy deals: %w", err) + } + + inSealingSubsystem := make(map[fsm.StateKey]struct{}, len(legacytypes.StatesKnownBySealingSubsystem)) + for _, s := range legacytypes.StatesKnownBySealingSubsystem { + inSealingSubsystem[s] = struct{}{} + } + + expiredStates := make(map[fsm.StateKey]struct{}, len(legacytypes.ProviderFinalityStates)) + for _, s := range legacytypes.ProviderFinalityStates { + expiredStates[s] = struct{}{} } + shards := make(map[string]struct{}) + var nSuccess int + var merr error + + for _, d := range legacyDeals { + // only announce deals that have been handed off to the sealing subsystem as the rest will get announced anyways + if _, ok := inSealingSubsystem[d.State]; !ok { + continue + } + // only announce deals that have not expired + if _, ok := expiredStates[d.State]; ok { + continue + } + + adCid, lerr := w.AnnounceLegcayDealToIndexer(ctx, d.ProposalCid) + if lerr != nil { + merr = multierror.Append(merr, lerr) + log.Errorw("failed to announce deal to Index provider", "proposalCid", d.ProposalCid, "err", lerr) + continue + } + log.Infof("announce legacy deal with proposal CID %s to the indexer with announcement-cid: %s", d.ProposalCid.String(), adCid.String()) + shards[d.Proposal.PieceCID.String()] = struct{}{} + nSuccess++ + } + + log.Infow("finished announcing active deals to index provider", "number of deals", nSuccess, "number of shards", shards) + log.Info("announcing all Boost deals to Indexer") deals, err := w.dealsDB.ListActive(ctx) if err != nil { return fmt.Errorf("failed to list deals: %w", err) } - shards := make(map[string]struct{}) - var nSuccess int - var merr error + bshards := make(map[string]struct{}) + var bnSuccess int for _, d := range deals { // filter out deals that will announce automatically at a later @@ -484,11 +515,11 @@ func (w *Wrapper) IndexerAnnounceAllDeals(ctx context.Context) error { } continue } - shards[d.ClientDealProposal.Proposal.PieceCID.String()] = struct{}{} - nSuccess++ + bshards[d.ClientDealProposal.Proposal.PieceCID.String()] = struct{}{} + bnSuccess++ } - log.Infow("finished announcing all boost deals to Indexer", "number of deals", nSuccess, "number of shards", len(shards)) + log.Infow("finished announcing all boost deals to Indexer", "number of deals", bnSuccess, "number of shards", len(bshards)) return merr } @@ -598,7 +629,7 @@ func (w *Wrapper) MultihashLister(ctx context.Context, prov peer.ID, contextID [ } // Deal was not found in boost DB - check in legacy markets - md, legacyErr := w.legacyProv.GetLocalDeal(proposalCid) + md, legacyErr := w.legacyProv.ByPropCid(proposalCid) if legacyErr == nil { // Found the deal, get an interator over the piece return provideF(proposalCid.String(), false, md.Proposal.PieceCID) @@ -716,7 +747,23 @@ type basicDealInfo struct { AnnounceToIPNI bool DealID string SectorID abi.SectorID - DealProposal storagemarket.ClientDealProposal + DealProposal legacytypes.ClientDealProposal +} + +func (w *Wrapper) AnnounceLegcayDealToIndexer(ctx context.Context, proposalCid cid.Cid) (cid.Cid, error) { + var deal legacytypes.MinerDeal + deal, err := w.legacyProv.ByPropCid(proposalCid) + if err != nil { + return cid.Undef, fmt.Errorf("failed getting deal %s: %w", proposalCid, err) + } + + mt := metadata.GraphsyncFilecoinV1{ + PieceCID: deal.Proposal.PieceCID, + FastRetrieval: deal.FastRetrieval, + VerifiedDeal: deal.Proposal.VerifiedDeal, + } + + return w.AnnounceBoostDealMetadata(ctx, mt, proposalCid.Bytes()) } func (w *Wrapper) AnnounceBoostDirectDeal(ctx context.Context, entry *types.DirectDeal) (cid.Cid, error) { diff --git a/indexprovider/wrapper_test.go b/indexprovider/wrapper_test.go index 713dd1fc6..8ab82a2e3 100644 --- a/indexprovider/wrapper_test.go +++ b/indexprovider/wrapper_test.go @@ -4,10 +4,11 @@ import ( "context" "testing" - "github.com/filecoin-project/boost-gfm/storagemarket" "github.com/filecoin-project/boost/db" "github.com/filecoin-project/boost/db/migrations" - "github.com/filecoin-project/boost/indexprovider/mock" + _ "github.com/filecoin-project/boost/lib/legacy/mocks" + mocks_legacy "github.com/filecoin-project/boost/lib/legacy/mocks" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" "github.com/filecoin-project/go-address" "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/go-state-types/builtin/v9/market" @@ -21,7 +22,7 @@ import ( func TestWrapperEmptyStorageListAndNoUpdates(t *testing.T) { wrapper, legacyStorageProvider, _, _ := setup(t) - legacyStorageProvider.EXPECT().ListLocalDeals().AnyTimes().Return(nil, nil) + legacyStorageProvider.EXPECT().ListDeals().AnyTimes().Return(nil, nil) // handleUpdates with an empty response from MinerAPI.StorageList() and no updates err := wrapper.handleUpdates(context.Background(), nil) @@ -52,7 +53,7 @@ func TestSectorStateManagerMatchingDealOnly(t *testing.T) { t.Run("deal in boost db", func(t *testing.T) { wrapper, legacyStorageProvider, storageMiner, prov := setup(t) - legacyStorageProvider.EXPECT().ListLocalDeals().Return(nil, nil) + legacyStorageProvider.EXPECT().ListDeals().Return(nil, nil) // Add a deal to the database deals, err := db.GenerateNDeals(1) @@ -73,11 +74,11 @@ func TestSectorStateManagerMatchingDealOnly(t *testing.T) { require.NoError(t, err) sectorNum := abi.SectorNumber(10) - deals := []storagemarket.MinerDeal{{ + deals := []legacytypes.MinerDeal{{ ClientDealProposal: boostDeals[0].ClientDealProposal, SectorNumber: sectorNum, }} - legacyStorageProvider.EXPECT().ListLocalDeals().Return(deals, nil) + legacyStorageProvider.EXPECT().ListDeals().Return(deals, nil) provAddr := deals[0].ClientDealProposal.Proposal.Provider runTest(t, wrapper, storageMiner, prov, provAddr, sectorNum) @@ -266,7 +267,7 @@ func TestSectorStateManagerStateChangeToIndexer(t *testing.T) { for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { wrapper, legacyStorageProvider, storageMiner, prov := setup(t) - legacyStorageProvider.EXPECT().ListLocalDeals().AnyTimes().Return(nil, nil) + legacyStorageProvider.EXPECT().ListDeals().AnyTimes().Return(nil, nil) // Add a deal to the database deals, err := db.GenerateNDeals(1) @@ -295,7 +296,7 @@ func TestSectorStateManagerStateChangeToIndexer(t *testing.T) { } } -func setup(t *testing.T) (*Wrapper, *mock.MockStorageProvider, *mockApiStorageMiner, *mock_provider.MockInterface) { +func setup(t *testing.T) (*Wrapper, *mocks_legacy.MockLegacyDealManager, *mockApiStorageMiner, *mock_provider.MockInterface) { ctx := context.Background() ctrl := gomock.NewController(t) prov := mock_provider.NewMockInterface(ctrl) @@ -304,19 +305,20 @@ func setup(t *testing.T) (*Wrapper, *mock.MockStorageProvider, *mockApiStorageMi require.NoError(t, db.CreateAllBoostTables(ctx, sqldb, sqldb)) require.NoError(t, migrations.Migrate(sqldb)) + legacyProv := mocks_legacy.NewMockLegacyDealManager(ctrl) + dealsDB := db.NewDealsDB(sqldb) storageMiner := &mockApiStorageMiner{} - storageProvider := mock.NewMockStorageProvider(ctrl) wrapper := &Wrapper{ enabled: true, dealsDB: dealsDB, prov: prov, - legacyProv: storageProvider, + legacyProv: legacyProv, meshCreator: &meshCreatorStub{}, } - return wrapper, storageProvider, storageMiner, prov + return wrapper, legacyProv, storageMiner, prov } type mockApiStorageMiner struct { diff --git a/itests/data_segment_index_retrieval_test.go b/itests/data_segment_index_retrieval_test.go index fe8c122f2..e826aba5c 100644 --- a/itests/data_segment_index_retrieval_test.go +++ b/itests/data_segment_index_retrieval_test.go @@ -13,6 +13,7 @@ import ( "github.com/google/uuid" "github.com/ipfs/go-cid" carv2 "github.com/ipld/go-car/v2" + trustlessutils "github.com/ipld/go-trustless-utils" "github.com/stretchr/testify/require" ) @@ -23,7 +24,6 @@ func TestDataSegmentIndexRetrieval(t *testing.T) { kit.QuietMiningLogs() framework.SetLogLevel() var opts []framework.FrameworkOpts - opts = append(opts, framework.EnableLegacyDeals(true)) opts = append(opts, framework.SetMaxStagingBytes(10000000)) // 10 MB f := framework.NewTestFramework(ctx, t, opts...) err := f.Start() @@ -39,21 +39,6 @@ func TestDataSegmentIndexRetrieval(t *testing.T) { err = f.AddClientProviderBalance(abi.NewTokenAmount(1e15)) require.NoError(t, err) - //// Create a CAR file - //tempdir := t.TempDir() - //log.Debugw("using tempdir", "dir", tempdir) - // - //// Select the number of car segments to use in test - //seg := 2 - // - //// Generate car file containing multiple car files - //segmentDetails, err := framework.GenerateDataSegmentFiles(t, tempdir, seg) - //require.NoError(t, err) - // - //p := segmentDetails.Piece.PieceCID.String() - // - //log.Info(p) - // Start a web server to serve the car files log.Debug("starting webserver") server, err := testutil.HttpTestFileServer(t, "fixtures") @@ -78,27 +63,19 @@ func TestDataSegmentIndexRetrieval(t *testing.T) { err = f.WaitForDealAddedToSector(dealUuid) require.NoError(t, err) - ////Retrieve and compare the all car files within the deal - //for i := 0; i < seg; i++ { - // for _, r := range segmentDetails.Segments[i].Root { - // outFile := f.RetrieveDirect(ctx, t, r, &res.DealParams.ClientDealProposal.Proposal.PieceCID, true) - // kit.AssertFilesEqual(t, segmentDetails.Segments[i].FilePath, outFile) - // } - //} - r1, err := cid.Parse("bafykbzaceaqliwrg6y2bxrhhbbiz3nknhz43yj2bqog4rulu5km5qhkckffuw") require.NoError(t, err) r2, err := cid.Parse("bafykbzaceccq64xf6yadlbmqpfindtf5x3cssel2fozkhvdyrrtnjnutr5j52") require.NoError(t, err) - outF1 := f.RetrieveDirect(ctx, t, r1, &pieceCid, false, nil) + outF1 := f.Retrieve(ctx, t, trustlessutils.Request{Root: r1, Scope: trustlessutils.DagScopeAll}, false) r, err := carv2.OpenReader(outF1) require.NoError(t, err) rs, err := r.Roots() require.NoError(t, err) require.Equal(t, r1, rs[0]) r.Close() - outf2 := f.RetrieveDirect(ctx, t, r2, &pieceCid, false, nil) + outf2 := f.Retrieve(ctx, t, trustlessutils.Request{Root: r2, Scope: trustlessutils.DagScopeAll}, false) r, err = carv2.OpenReader(outf2) require.NoError(t, err) rs, err = r.Roots() diff --git a/itests/disabled_markets_v1_deal_test.go b/itests/disabled_markets_v1_deal_test.go deleted file mode 100644 index 566504bb6..000000000 --- a/itests/disabled_markets_v1_deal_test.go +++ /dev/null @@ -1,53 +0,0 @@ -package itests - -import ( - "context" - "testing" - - "github.com/filecoin-project/boost/itests/framework" - "github.com/filecoin-project/boost/testutil" - lapi "github.com/filecoin-project/lotus/api" - "github.com/filecoin-project/lotus/itests/kit" - "github.com/stretchr/testify/require" -) - -func TestDisabledMarketsV1Deal(t *testing.T) { - ctx := context.Background() - log := framework.Log - - kit.QuietMiningLogs() - framework.SetLogLevel() - var opts []framework.FrameworkOpts - opts = append(opts, framework.EnableLegacyDeals(false)) - f := framework.NewTestFramework(ctx, t, opts...) - err := f.Start() - require.NoError(t, err) - defer f.Stop() - - // Create a CAR file - log.Debugw("using tempdir", "dir", f.HomeDir) - rseed := 0 - size := 7 << 20 // 7MiB file - - inPath, err := testutil.CreateRandomFile(f.HomeDir, rseed, size) - require.NoError(t, err) - res, err := f.FullNode.ClientImport(ctx, lapi.FileRef{Path: inPath}) - require.NoError(t, err) - - // Create a new markets v1 deal - dp := f.DefaultMarketsV1DealParams() - dp.Data.Root = res.Root - - log.Debugw("starting deal", "root", res.Root) - dealProposalCid, err := f.FullNode.ClientStartDeal(ctx, &dp) - require.NoError(t, err) - - log.Debugw("got deal proposal cid", "cid", dealProposalCid) - di, err := f.FullNode.ClientGetDealInfo(ctx, *dealProposalCid) - require.NoError(t, err) - - log.Debugw(di.Message) - - err = f.WaitDealSealed(ctx, dealProposalCid) - require.ErrorContains(t, err, "protocol are deprecated") -} diff --git a/itests/dummydeal_offline_test.go b/itests/dummydeal_offline_test.go index 4abf1ceac..0327fccab 100644 --- a/itests/dummydeal_offline_test.go +++ b/itests/dummydeal_offline_test.go @@ -9,6 +9,7 @@ import ( "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/lotus/itests/kit" "github.com/google/uuid" + trustlessutils "github.com/ipld/go-trustless-utils" "github.com/stretchr/testify/require" ) @@ -18,7 +19,6 @@ func TestDummydealOffline(t *testing.T) { kit.QuietMiningLogs() framework.SetLogLevel() var opts []framework.FrameworkOpts - opts = append(opts, framework.EnableLegacyDeals(true)) f := framework.NewTestFramework(ctx, t, opts...) err := f.Start() require.NoError(t, err) @@ -45,4 +45,12 @@ func TestDummydealOffline(t *testing.T) { require.True(t, res.Accepted) err = f.WaitForDealAddedToSector(offlineDealUuid) require.NoError(t, err) + + outFile := f.Retrieve( + ctx, + t, + trustlessutils.Request{Root: rootCid, Scope: trustlessutils.DagScopeAll}, + true, + ) + kit.AssertFilesEqual(t, randomFilepath, outFile) } diff --git a/itests/dummydeal_podsi_test.go b/itests/dummydeal_podsi_test.go index 2a9a3ccb6..a1216e78e 100644 --- a/itests/dummydeal_podsi_test.go +++ b/itests/dummydeal_podsi_test.go @@ -30,7 +30,7 @@ import ( "github.com/ipld/go-ipld-prime" cidlink "github.com/ipld/go-ipld-prime/linking/cid" "github.com/multiformats/go-multicodec" - multihash "github.com/multiformats/go-multihash" + "github.com/multiformats/go-multihash" "github.com/stretchr/testify/require" ) @@ -43,7 +43,7 @@ func TestDummyPodsiDealOnline(t *testing.T) { kit.QuietMiningLogs() framework.SetLogLevel() var opts []framework.FrameworkOpts - opts = append(opts, framework.EnableLegacyDeals(true), framework.SetMaxStagingBytes(10e9), framework.SetProvisionalWalletBalances(9e18)) + opts = append(opts, framework.SetMaxStagingBytes(10e9), framework.SetProvisionalWalletBalances(9e18)) f := framework.NewTestFramework(ctx, t, opts...) err := f.Start() require.NoError(t, err) diff --git a/itests/dummydeal_test.go b/itests/dummydeal_test.go index 1de5afeb3..607882157 100644 --- a/itests/dummydeal_test.go +++ b/itests/dummydeal_test.go @@ -12,6 +12,7 @@ import ( "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/lotus/itests/kit" "github.com/google/uuid" + trustlessutils "github.com/ipld/go-trustless-utils" "github.com/stretchr/testify/require" ) @@ -22,7 +23,6 @@ func TestDummydealOnline(t *testing.T) { kit.QuietMiningLogs() framework.SetLogLevel() var opts []framework.FrameworkOpts - opts = append(opts, framework.EnableLegacyDeals(true)) f := framework.NewTestFramework(ctx, t, opts...) err := f.Start() require.NoError(t, err) @@ -94,6 +94,11 @@ func TestDummydealOnline(t *testing.T) { require.NoError(t, err) // rootCid is an identity CID - outFile := f.RetrieveDirect(ctx, t, rootCid, &res.DealParams.ClientDealProposal.Proposal.PieceCID, true, nil) + outFile := f.Retrieve( + ctx, + t, + trustlessutils.Request{Root: rootCid, Scope: trustlessutils.DagScopeAll}, + true, + ) kit.AssertFilesEqual(t, randomFilepath, outFile) } diff --git a/itests/framework/fixtures.go b/itests/framework/fixtures.go deleted file mode 100644 index 1aaa8df82..000000000 --- a/itests/framework/fixtures.go +++ /dev/null @@ -1,164 +0,0 @@ -package framework - -import ( - "errors" - "io" - "math/bits" - "os" - "testing" - - "github.com/filecoin-project/boost/storagemarket" - "github.com/filecoin-project/go-data-segment/datasegment" - commcid "github.com/filecoin-project/go-fil-commcid" - commp "github.com/filecoin-project/go-fil-commp-hashhash" - "github.com/filecoin-project/go-state-types/abi" - "github.com/filecoin-project/lotus/itests/kit" - "github.com/ipfs/go-cid" - "github.com/ipld/go-car/v2" -) - -type CarDetails struct { - CarPath string - Root []cid.Cid - FilePath string -} - -type SegmentDetails struct { - Piece *abi.PieceInfo - Segments []*CarDetails - CarPath string - CarSize int64 -} - -func GenerateDataSegmentFiles(t *testing.T, tmpdir string, num int) (SegmentDetails, error) { - if num < 2 { - return SegmentDetails{}, errors.New("at least 2 deals are required to test data segment index") - } - - fileSize := 1572864 - - var cars []*CarDetails - for i := 1; i <= num; i++ { - - carPath, filePath := kit.CreateRandomCARv1(t, i, fileSize) - rd, err := car.OpenReader(carPath) - if err != nil { - return SegmentDetails{}, err - } - - roots, err := rd.Roots() - if err != nil { - return SegmentDetails{}, err - } - - err = rd.Close() - if err != nil { - return SegmentDetails{}, err - } - - cars = append(cars, &CarDetails{ - CarPath: carPath, - FilePath: filePath, - Root: roots, - }) - } - - finalCar, err := os.CreateTemp(tmpdir, "finalcar") - if err != nil { - return SegmentDetails{}, err - } - - err = generateDataSegmentCar(cars, finalCar) - if err != nil { - return SegmentDetails{}, err - } - - finalCarName := finalCar.Name() - carStat, err := finalCar.Stat() - if err != nil { - return SegmentDetails{}, err - } - carSize := carStat.Size() - err = finalCar.Close() - if err != nil { - return SegmentDetails{}, err - } - - cidAndSize, err := storagemarket.GenerateCommPLocally(finalCarName) - if err != nil { - return SegmentDetails{}, err - } - - return SegmentDetails{ - Piece: cidAndSize, - Segments: cars, - CarPath: finalCarName, - CarSize: carSize, - }, nil -} - -func generateDataSegmentCar(cars []*CarDetails, outputFile *os.File) error { - - readers := make([]io.Reader, 0) - deals := make([]abi.PieceInfo, 0) - - for _, cf := range cars { - - r, err := os.Open(cf.CarPath) - - if err != nil { - return err - } - - readers = append(readers, r) - cp := new(commp.Calc) - - _, err = io.Copy(cp, r) - if err != nil { - return err - } - - rawCommP, size, err := cp.Digest() - if err != nil { - return err - } - - _, err = r.Seek(0, io.SeekStart) - if err != nil { - return err - } - - c, _ := commcid.DataCommitmentV1ToCID(rawCommP) - - subdeal := abi.PieceInfo{ - Size: abi.PaddedPieceSize(size), - PieceCID: c, - } - deals = append(deals, subdeal) - } - - _, size, err := datasegment.ComputeDealPlacement(deals) - if err != nil { - return err - } - - overallSize := abi.PaddedPieceSize(size) - // we need to make this the 'next' power of 2 in order to have space for the index - next := 1 << (64 - bits.LeadingZeros64(uint64(overallSize+256))) - - a, err := datasegment.NewAggregate(abi.PaddedPieceSize(next), deals) - if err != nil { - return err - } - out, err := a.AggregateObjectReader(readers) - if err != nil { - return err - } - - _, err = io.Copy(outputFile, out) - if err != nil { - return err - } - - return nil -} diff --git a/itests/framework/framework.go b/itests/framework/framework.go index f7bb624ff..439bd712a 100644 --- a/itests/framework/framework.go +++ b/itests/framework/framework.go @@ -1,33 +1,40 @@ package framework import ( + "bytes" "context" "encoding/json" "errors" "fmt" + "io" "math/rand" "os" "path" + "strings" "sync" "testing" "time" - gfm_storagemarket "github.com/filecoin-project/boost-gfm/storagemarket" - "github.com/filecoin-project/boost-gfm/storagemarket/impl/storedask" + "github.com/dustin/go-humanize" "github.com/filecoin-project/boost/api" + clinode "github.com/filecoin-project/boost/cli/node" boostclient "github.com/filecoin-project/boost/client" + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/markets/utils" "github.com/filecoin-project/boost/node" "github.com/filecoin-project/boost/node/config" "github.com/filecoin-project/boost/node/modules/dtypes" "github.com/filecoin-project/boost/node/repo" + rc "github.com/filecoin-project/boost/retrievalmarket/client" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" "github.com/filecoin-project/boost/storagemarket" + "github.com/filecoin-project/boost/storagemarket/storedask" "github.com/filecoin-project/boost/storagemarket/types" "github.com/filecoin-project/boost/storagemarket/types/dealcheckpoints" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" types2 "github.com/filecoin-project/boost/transport/types" "github.com/filecoin-project/go-address" cborutil "github.com/filecoin-project/go-cbor-util" - lotus_gfm_retrievalmarket "github.com/filecoin-project/go-fil-markets/retrievalmarket" - lotus_gfm_storagemarket "github.com/filecoin-project/go-fil-markets/storagemarket" "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/go-state-types/big" "github.com/filecoin-project/go-state-types/builtin" @@ -36,7 +43,6 @@ import ( "github.com/filecoin-project/go-state-types/exitcode" lapi "github.com/filecoin-project/lotus/api" "github.com/filecoin-project/lotus/api/v1api" - lbuild "github.com/filecoin-project/lotus/build" "github.com/filecoin-project/lotus/chain/actors" chaintypes "github.com/filecoin-project/lotus/chain/types" ltypes "github.com/filecoin-project/lotus/chain/types" @@ -52,6 +58,9 @@ import ( "github.com/filecoin-project/lotus/storage/pipeline/sealiface" "github.com/filecoin-project/lotus/storage/sealer/storiface" "github.com/google/uuid" + "github.com/ipfs/boxo/blockservice" + "github.com/ipfs/boxo/blockstore" + "github.com/ipfs/boxo/exchange/offline" "github.com/ipfs/boxo/files" dag "github.com/ipfs/boxo/ipld/merkledag" dstest "github.com/ipfs/boxo/ipld/merkledag/test" @@ -59,26 +68,35 @@ import ( blocks "github.com/ipfs/go-block-format" "github.com/ipfs/go-cid" "github.com/ipfs/go-datastore" + flatfs "github.com/ipfs/go-ds-flatfs" + levelds "github.com/ipfs/go-ds-leveldb" ipldcbor "github.com/ipfs/go-ipld-cbor" ipldformat "github.com/ipfs/go-ipld-format" logging "github.com/ipfs/go-log/v2" "github.com/ipld/go-car" + carv2 "github.com/ipld/go-car/v2" + storagecar "github.com/ipld/go-car/v2/storage" "github.com/ipld/go-ipld-prime" "github.com/ipld/go-ipld-prime/codec/dagjson" "github.com/ipld/go-ipld-prime/datamodel" - "github.com/ipld/go-ipld-prime/traversal/selector" + "github.com/ipld/go-ipld-prime/linking" + cidlink "github.com/ipld/go-ipld-prime/linking/cid" + trustless "github.com/ipld/go-trustless-utils" + "github.com/ipld/go-trustless-utils/traversal" "github.com/libp2p/go-libp2p" "github.com/libp2p/go-libp2p/core/host" + "github.com/libp2p/go-libp2p/core/peer" + "github.com/libp2p/go-libp2p/core/peerstore" "github.com/multiformats/go-multihash" "github.com/stretchr/testify/require" "golang.org/x/sync/errgroup" + "golang.org/x/term" ) var Log = logging.Logger("boosttest") type TestFrameworkConfig struct { Ensemble *kit.Ensemble - EnableLegacy bool MaxStagingBytes int64 ProvisionalWalletBalances int64 } @@ -100,12 +118,6 @@ type TestFramework struct { type FrameworkOpts func(pc *TestFrameworkConfig) -func EnableLegacyDeals(enable bool) FrameworkOpts { - return func(tmc *TestFrameworkConfig) { - tmc.EnableLegacy = enable - } -} - func SetMaxStagingBytes(max int64) FrameworkOpts { return func(tmc *TestFrameworkConfig) { tmc.MaxStagingBytes = max @@ -118,6 +130,12 @@ func WithEnsemble(e *kit.Ensemble) FrameworkOpts { } } +func WithMaxStagingDealsBytes(e int64) FrameworkOpts { + return func(tmc *TestFrameworkConfig) { + tmc.MaxStagingBytes = e + } +} + func SetProvisionalWalletBalances(balance int64) FrameworkOpts { return func(tmc *TestFrameworkConfig) { tmc.ProvisionalWalletBalances = balance @@ -337,21 +355,18 @@ func (f *TestFramework) Start(opts ...ConfigOpt) error { cfg.Wallets.Miner = minerAddr.String() cfg.Wallets.PublishStorageDeals = psdWalletAddr.String() cfg.Wallets.DealCollateral = dealCollatAddr.String() - cfg.LotusDealmaking.MaxDealsPerPublishMsg = 1 - cfg.LotusDealmaking.PublishMsgPeriod = lotus_config.Duration(0) + cfg.Dealpublish.MaxDealsPerPublishMsg = 1 + cfg.Dealpublish.PublishMsgPeriod = config.Duration(0) val, err := ltypes.ParseFIL("0.1 FIL") if err != nil { return err } - cfg.LotusFees.MaxPublishDealsFee = val + cfg.Dealpublish.MaxPublishDealsFee = val cfg.Dealmaking.RemoteCommp = true // No transfers will start until the first stall check period has elapsed - cfg.Dealmaking.HttpTransferStallCheckPeriod = config.Duration(100 * time.Millisecond) + cfg.HttpDownload.HttpTransferStallCheckPeriod = config.Duration(100 * time.Millisecond) cfg.Storage.ParallelFetchLimit = 10 - if f.config.EnableLegacy { - cfg.Dealmaking.EnableLegacyStorageDeals = true - } for _, o := range opts { o(cfg) @@ -683,18 +698,6 @@ func (f *TestFramework) signProposal(addr address.Address, proposal *market.Deal }, nil } -func (f *TestFramework) DefaultMarketsV1DealParams() lapi.StartDealParams { - return lapi.StartDealParams{ - Data: &lotus_gfm_storagemarket.DataRef{TransferType: gfm_storagemarket.TTGraphsync}, - EpochPrice: ltypes.NewInt(62500000), // minimum asking price - MinBlocksDuration: uint64(lbuild.MinDealDuration), - Miner: f.MinerAddr, - Wallet: f.DefaultWallet, - DealStartEpoch: 35000, - FastRetrieval: true, - } -} - func sendFunds(ctx context.Context, sender lapi.FullNode, recipient address.Address, amount abi.TokenAmount) error { senderAddr, err := sender.WalletDefaultAddress(ctx) if err != nil { @@ -763,14 +766,14 @@ func (f *TestFramework) WaitDealSealed(ctx context.Context, deal *cid.Cid) error } switch di.State { - case gfm_storagemarket.StorageDealAwaitingPreCommit, gfm_storagemarket.StorageDealSealing: - case gfm_storagemarket.StorageDealProposalRejected: + case legacytypes.StorageDealAwaitingPreCommit, legacytypes.StorageDealSealing: + case legacytypes.StorageDealProposalRejected: return errors.New("deal rejected") - case gfm_storagemarket.StorageDealFailing: + case legacytypes.StorageDealFailing: return errors.New("deal failed") - case gfm_storagemarket.StorageDealError: + case legacytypes.StorageDealError: return fmt.Errorf("deal errored: %s", di.Message) - case gfm_storagemarket.StorageDealActive: + case legacytypes.StorageDealActive: return nil } @@ -778,18 +781,6 @@ func (f *TestFramework) WaitDealSealed(ctx context.Context, deal *cid.Cid) error } } -func (f *TestFramework) Retrieve(ctx context.Context, t *testing.T, deal *cid.Cid, root cid.Cid, extractCar bool, selectorNode datamodel.Node) string { - // perform retrieval. - info, err := f.FullNode.ClientGetDealInfo(ctx, *deal) - require.NoError(t, err) - - offers, err := f.FullNode.ClientFindData(ctx, root, &info.PieceCID) - require.NoError(t, err) - require.NotEmpty(t, offers, "no offers") - - return f.retrieve(ctx, t, offers[0], extractCar, selectorNode) -} - func (f *TestFramework) ExtractFileFromCAR(ctx context.Context, t *testing.T, file *os.File) string { bserv := dstest.Bserv() ch, err := car.LoadCar(ctx, bserv.Blockstore(), file) @@ -825,79 +816,207 @@ func (f *TestFramework) ExtractFileFromCAR(ctx context.Context, t *testing.T, fi return tmpFile } -func (f *TestFramework) RetrieveDirect(ctx context.Context, t *testing.T, root cid.Cid, pieceCid *cid.Cid, extractCar bool, selectorNode datamodel.Node) string { - offer, err := f.FullNode.ClientMinerQueryOffer(ctx, f.MinerAddr, root, pieceCid) +func (f *TestFramework) Retrieve(ctx context.Context, t *testing.T, request trustless.Request, extractCar bool) string { + tempdir := t.TempDir() + + var out string + retPath := path.Join(tempdir, "retrievals") + require.NoError(t, os.Mkdir(retPath, 0755)) + + clientPath := path.Join(tempdir, "client") + require.NoError(t, os.Mkdir(clientPath, 0755)) + + clientNode, err := clinode.Setup(clientPath) require.NoError(t, err) - return f.retrieve(ctx, t, offer, extractCar, selectorNode) -} + addr, err := clientNode.Wallet.GetDefault() + require.NoError(t, err) -func (f *TestFramework) retrieve(ctx context.Context, t *testing.T, offer lapi.QueryOffer, extractCar bool, selectorNode datamodel.Node) string { - p := path.Join(t.TempDir(), "ret-car-"+t.Name()) - err := os.MkdirAll(path.Dir(p), 0755) + bstoreDatastore, err := flatfs.CreateOrOpen(path.Join(tempdir, "blockstore"), flatfs.NextToLast(3), false) + bstore := blockstore.NewBlockstore(bstoreDatastore, blockstore.NoPrefix()) require.NoError(t, err) - carFile, err := os.Create(p) + + ds, err := levelds.NewDatastore(path.Join(clientPath, "dstore"), nil) + require.NoError(t, err) + + // Create the retrieval client + fc, err := rc.NewClient(clientNode.Host, f.FullNode, clientNode.Wallet, addr, bstore, ds, clientPath) + require.NoError(t, err) + + baddrs, err := f.Boost.NetAddrsListen(ctx) require.NoError(t, err) - defer carFile.Close() //nolint:errcheck + // Query the remote to find out the retrieval parameters + query, err := RetrievalQuery(ctx, t, clientNode, &baddrs, request.Root) + require.NoError(t, err) - caddr, err := f.FullNode.WalletDefaultAddress(ctx) + // Create a matching proposal for the query + proposal, err := rc.RetrievalProposalForAsk(query, request.Root, request.Selector()) require.NoError(t, err) - updatesCtx, cancel := context.WithCancel(ctx) - updates, err := f.FullNode.ClientGetRetrievalUpdates(updatesCtx) + // Let's see the selector we're working with + encoded, err := ipld.Encode(request.Selector(), dagjson.Encode) require.NoError(t, err) + t.Logf("Retrieving with selector: %s", string(encoded)) + + // Retrieve the data + _, err = fc.RetrieveContentWithProgressCallback( + ctx, + f.MinerAddr, + proposal, + func(bytesReceived_ uint64) { + printProgress(bytesReceived_, t) + }, + ) + require.NoError(t, err) + + // Validate the data - order := offer.Order(caddr) - if selectorNode != nil { - _, err := selector.CompileSelector(selectorNode) + dservOffline := dag.NewDAGService(blockservice.New(bstore, offline.Exchange(bstore))) + lsys := utils.CreateLinkSystem(dservOffline) + + if !extractCar { + // If the caller wants a CAR, we create it and then when we run our check traversal over the DAG + // each load will trigger a write to the CAR + file, err := os.CreateTemp(retPath, "*"+request.Root.String()+".car") require.NoError(t, err) - jsonSelector, err := ipld.Encode(selectorNode, dagjson.Encode) + out = file.Name() + storage, err := storagecar.NewWritable(file, []cid.Cid{request.Root}, carv2.WriteAsCarV1(true)) require.NoError(t, err) - sel := lapi.Selector(jsonSelector) - order.DataSelector = &sel + sro := lsys.StorageReadOpener + lsys.StorageReadOpener = func(lc linking.LinkContext, l datamodel.Link) (io.Reader, error) { + r, err := sro(lc, l) + if err != nil { + return nil, err + } + buf, err := io.ReadAll(r) + if err != nil { + return nil, err + } + if err := storage.Put(lc.Ctx, l.(cidlink.Link).Cid.KeyString(), buf); err != nil { + return nil, err + } + return bytes.NewReader(buf), nil + } } - retrievalRes, err := f.FullNode.ClientRetrieve(ctx, order) + // Check that we got what we expected by executing the same selector over our + // retrieved DAG + _, err = traversal.Config{ + Root: request.Root, + Selector: request.Selector(), + }.Traverse(ctx, lsys, nil) require.NoError(t, err) -consumeEvents: - for { - var evt lapi.RetrievalInfo - select { - case <-updatesCtx.Done(): - t.Fatal("Retrieval Timed Out") - case evt = <-updates: - if evt.ID != retrievalRes.DealID { - continue - } - } - switch evt.Status { - case lotus_gfm_retrievalmarket.DealStatusCompleted: - break consumeEvents - case lotus_gfm_retrievalmarket.DealStatusRejected: - t.Fatalf("Retrieval Proposal Rejected: %s", evt.Message) - case - lotus_gfm_retrievalmarket.DealStatusDealNotFound, - lotus_gfm_retrievalmarket.DealStatusErrored: - t.Fatalf("Retrieval Error: %s", evt.Message) + + if extractCar { + // Caller doesn't want the raw blocks, so extract the file as UnixFS and + // assume that we've fetched the right blocks to be able to do this. + dnode, err := dservOffline.Get(ctx, request.Root) + require.NoError(t, err) + ufsFile, err := unixfile.NewUnixfsFile(ctx, dservOffline, dnode) + require.NoError(t, err) + file, err := os.CreateTemp(retPath, "*"+request.Root.String()) + require.NoError(t, err) + err = file.Close() + require.NoError(t, err) + err = os.Remove(file.Name()) + require.NoError(t, err) + err = files.WriteTo(ufsFile, file.Name()) + require.NoError(t, err) + out = file.Name() + } + + return out +} + +type RetrievalInfo struct { + PayloadCID cid.Cid + ID legacyretrievaltypes.DealID + PieceCID *cid.Cid + PricePerByte abi.TokenAmount + UnsealPrice abi.TokenAmount + + Status legacyretrievaltypes.DealStatus + Message string // more information about deal state, particularly errors + Provider peer.ID + BytesReceived uint64 + BytesPaidFor uint64 + TotalPaid abi.TokenAmount + + TransferChannelID *datatransfer.ChannelID + DataTransfer *DataTransferChannel + + // optional event if part of ClientGetRetrievalUpdates + Event *legacyretrievaltypes.ClientEvent +} + +type RestrievalRes struct { + DealID legacyretrievaltypes.DealID +} + +type DataTransferChannel struct { + TransferID datatransfer.TransferID + Status datatransfer.Status + BaseCID cid.Cid + IsInitiator bool + IsSender bool + Voucher string + Message string + OtherPeer peer.ID + Transferred uint64 + Stages *datatransfer.ChannelStages +} + +func printProgress(bytesReceived uint64, t *testing.T) { + str := fmt.Sprintf("%v (%v)", bytesReceived, humanize.IBytes(bytesReceived)) + + termWidth, _, err := term.GetSize(int(os.Stdin.Fd())) + strLen := len(str) + if err == nil { + + if strLen < termWidth { + // If the string is shorter than the terminal width, pad right side + // with spaces to remove old text + str = strings.Join([]string{str, strings.Repeat(" ", termWidth-strLen)}, "") + } else if strLen > termWidth { + // If the string doesn't fit in the terminal, cut it down to a size + // that fits + str = str[:termWidth] } } - cancel() - require.NoError(t, f.FullNode.ClientExport(ctx, - lapi.ExportRef{ - Root: offer.Root, - DealID: retrievalRes.DealID, - }, - lapi.FileRef{ - Path: carFile.Name(), - IsCAR: true, - })) + t.Logf("%s\r", str) +} - ret := carFile.Name() - if extractCar { - ret = f.ExtractFileFromCAR(ctx, t, carFile) +func RetrievalQuery(ctx context.Context, t *testing.T, client *clinode.Node, peerAddr *peer.AddrInfo, pcid cid.Cid) (*legacyretrievaltypes.QueryResponse, error) { + client.Host.Peerstore().AddAddrs(peerAddr.ID, peerAddr.Addrs, peerstore.TempAddrTTL) + s, err := client.Host.NewStream(ctx, peerAddr.ID, rc.RetrievalQueryProtocol) + require.NoError(t, err) + + client.Host.ConnManager().Protect(s.Conn().RemotePeer(), "RetrievalQuery") + defer func() { + client.Host.ConnManager().Unprotect(s.Conn().RemotePeer(), "RetrievalQuery") + s.Close() + }() + + // We have connected + + q := &legacyretrievaltypes.Query{ + PayloadCID: pcid, + } + + var resp legacyretrievaltypes.QueryResponse + dline, ok := ctx.Deadline() + if ok { + _ = s.SetDeadline(dline) + defer func() { _ = s.SetDeadline(time.Time{}) }() } - return ret + err = cborutil.WriteCborRPC(s, q) + require.NoError(t, err) + + err = cborutil.ReadCborRPC(s, &resp) + require.NoError(t, err) + + return &resp, nil } diff --git a/itests/framework/log.go b/itests/framework/log.go index c6668bd6b..219b4927b 100644 --- a/itests/framework/log.go +++ b/itests/framework/log.go @@ -6,6 +6,9 @@ func SetLogLevel() { _ = logging.SetLogLevel("boosttest", "DEBUG") _ = logging.SetLogLevel("devnet", "DEBUG") _ = logging.SetLogLevel("boost", "DEBUG") + _ = logging.SetLogLevel("graphsync", "DEBUG") + _ = logging.SetLogLevel("boostgs", "DEBUG") + _ = logging.SetLogLevel("dt_graphsync", "DEBUG") _ = logging.SetLogLevel("provider", "DEBUG") _ = logging.SetLogLevel("http-transfer", "DEBUG") _ = logging.SetLogLevel("boost-provider", "DEBUG") diff --git a/itests/markets_v1_identity_cid_test.go b/itests/graphsync_identity_cid_test.go similarity index 68% rename from itests/markets_v1_identity_cid_test.go rename to itests/graphsync_identity_cid_test.go index fac41b268..efdbc2bad 100644 --- a/itests/markets_v1_identity_cid_test.go +++ b/itests/graphsync_identity_cid_test.go @@ -5,13 +5,17 @@ import ( "fmt" "math/rand" "os" + "path/filepath" "testing" + "time" + "github.com/davecgh/go-spew/spew" gstestutil "github.com/filecoin-project/boost-graphsync/testutil" "github.com/filecoin-project/boost/itests/framework" "github.com/filecoin-project/boost/testutil" - lapi "github.com/filecoin-project/lotus/api" + "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/lotus/itests/kit" + "github.com/google/uuid" "github.com/ipfs/go-cid" "github.com/ipld/go-car/v2" "github.com/ipld/go-car/v2/storage" @@ -20,12 +24,12 @@ import ( "github.com/ipld/go-ipld-prime/fluent/qp" cidlink "github.com/ipld/go-ipld-prime/linking/cid" "github.com/ipld/go-ipld-prime/node/basicnode" - selectorparse "github.com/ipld/go-ipld-prime/traversal/selector/parse" - multihash "github.com/multiformats/go-multihash" + trustlessutils "github.com/ipld/go-trustless-utils" + "github.com/multiformats/go-multihash" "github.com/stretchr/testify/require" ) -func TestMarketsV1DealAndRetrievalWithIdentityCID(t *testing.T) { +func TestDealAndRetrievalWithIdentityCID(t *testing.T) { req := require.New(t) ctx := context.Background() log := framework.Log @@ -33,14 +37,19 @@ func TestMarketsV1DealAndRetrievalWithIdentityCID(t *testing.T) { kit.QuietMiningLogs() framework.SetLogLevel() var opts []framework.FrameworkOpts - opts = append(opts, framework.EnableLegacyDeals(true)) + opts = append(opts, framework.WithMaxStagingDealsBytes(10000000)) f := framework.NewTestFramework(ctx, t, opts...) err := f.Start() req.NoError(err) defer f.Stop() + err = f.AddClientProviderBalance(abi.NewTokenAmount(1e15)) + require.NoError(t, err) + // Create a CAR file - carPath := f.HomeDir + "/testfile.car" + tempdir := t.TempDir() + log.Debugw("using tempdir", "dir", tempdir) + carPath := tempdir + "/testfile.car" log.Debugf("using test car %s", carPath) carFile, err := os.Create(carPath) req.NoError(err) @@ -82,29 +91,35 @@ func TestMarketsV1DealAndRetrievalWithIdentityCID(t *testing.T) { req.NoError(car.ReplaceRootsInFile(carPath, []cid.Cid{rootLink.(cidlink.Link).Cid})) log.Debugw("filled car, replaced root with correct root", "root", rootLink.String()) - // Import and make a deal to store - - res, err := f.FullNode.ClientImport(ctx, lapi.FileRef{Path: carPath, IsCAR: true}) - req.NoError(err) - - log.Debugw("imported data for deal") - - dp := f.DefaultMarketsV1DealParams() - dp.Data.Root = res.Root - - log.Debugw("starting deal", "root", res.Root) - dealProposalCid, err := f.FullNode.ClientStartDeal(ctx, &dp) - req.NoError(err) - - log.Debugw("got deal proposal cid", "cid", dealProposalCid) - - err = f.WaitDealSealed(ctx, dealProposalCid) - req.NoError(err) + // Start a web server to serve the car files + log.Debug("starting webserver") + server, err := testutil.HttpTestFileServer(t, tempdir) + require.NoError(t, err) + defer server.Close() + + // Create a new dummy deal + log.Debug("creating dummy deal") + dealUuid := uuid.New() + root := rootLink.(cidlink.Link).Cid + + // Make a deal + res, err := f.MakeDummyDeal(dealUuid, carPath, root, server.URL+"/"+filepath.Base(carPath), false) + require.NoError(t, err) + require.True(t, res.Result.Accepted) + log.Debugw("got response from MarketDummyDeal", "res", spew.Sdump(res)) + dealCid, err := res.DealParams.ClientDealProposal.Proposal.Cid() + require.NoError(t, err) + log.Infof("deal ID is : %s", dealCid.String()) + // Wait for the first deal to be added to a sector and cleaned up so space is made + err = f.WaitForDealAddedToSector(dealUuid) + require.NoError(t, err) + + time.Sleep(5 * time.Second) // Deal is stored and sealed, attempt different retrieval forms - log.Debugw("deal is sealed, starting retrieval", "cid", dealProposalCid, "root", res.Root) - outPath := f.Retrieve(ctx, t, dealProposalCid, rootLink.(cidlink.Link).Cid, false, selectorparse.CommonSelector_ExploreAllRecursively) + log.Debugw("deal is sealed, starting retrieval", "cid", dealCid.String(), "root", root.String()) + outPath := f.Retrieve(ctx, t, trustlessutils.Request{Root: root, Scope: trustlessutils.DagScopeAll}, false) // Inspect what we got gotCids, err := testutil.CidsInCar(outPath) diff --git a/itests/markets_v1_retrieval_test.go b/itests/graphsync_retrieval_test.go similarity index 61% rename from itests/markets_v1_retrieval_test.go rename to itests/graphsync_retrieval_test.go index 44cd6c2c1..7ec81cf71 100644 --- a/itests/markets_v1_retrieval_test.go +++ b/itests/graphsync_retrieval_test.go @@ -2,40 +2,44 @@ package itests import ( "context" - "math" + "path/filepath" "testing" + "time" + "github.com/davecgh/go-spew/spew" "github.com/filecoin-project/boost/itests/framework" "github.com/filecoin-project/boost/testutil" - lapi "github.com/filecoin-project/lotus/api" + "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/lotus/itests/kit" + "github.com/google/uuid" "github.com/ipfs/go-cid" - "github.com/ipfs/go-unixfsnode" - "github.com/ipld/go-ipld-prime/datamodel" - "github.com/ipld/go-ipld-prime/node/basicnode" - "github.com/ipld/go-ipld-prime/traversal/selector/builder" + trustless "github.com/ipld/go-trustless-utils" "github.com/stretchr/testify/require" ) -func TestMarketsV1DealRetrieval(t *testing.T) { +func TestDealRetrieval(t *testing.T) { ctx := context.Background() log := framework.Log kit.QuietMiningLogs() framework.SetLogLevel() var opts []framework.FrameworkOpts - opts = append(opts, framework.EnableLegacyDeals(true)) + opts = append(opts, framework.WithMaxStagingDealsBytes(10000000)) f := framework.NewTestFramework(ctx, t, opts...) err := f.Start() require.NoError(t, err) defer f.Stop() + err = f.AddClientProviderBalance(abi.NewTokenAmount(1e15)) + require.NoError(t, err) + // Create a CAR file - log.Debugw("using tempdir", "dir", f.HomeDir) + tempdir := t.TempDir() + log.Debugw("using tempdir", "dir", tempdir) rseed := 0 size := 7 << 20 // 7MiB file - inPath, dirEnt := testutil.CreateRandomUnixfsFileInCar(t, f.HomeDir, rseed, size) + inPath, dirEnt := testutil.CreateRandomUnixfsFileInCar(t, tempdir, rseed, size) root := dirEnt.Root leaves := dirEnt.SelfCids[:len(dirEnt.SelfCids)-1] /* @@ -77,71 +81,99 @@ func TestMarketsV1DealRetrieval(t *testing.T) { bafkreifokzy5zcluf3hj23nkrvr7tx6sivpshkd4be5tpfibk6vm2mzlxy | RawLeaf | /0[7172032:7340031] (168000 B) */ - // Import and make a deal to store - - res, err := f.FullNode.ClientImport(ctx, lapi.FileRef{Path: inPath, IsCAR: true}) + // Start a web server to serve the car files + log.Debug("starting webserver") + server, err := testutil.HttpTestFileServer(t, tempdir) require.NoError(t, err) + defer server.Close() - dp := f.DefaultMarketsV1DealParams() - dp.Data.Root = res.Root + // Create a new dummy deal + log.Debug("creating dummy deal") + dealUuid := uuid.New() - log.Debugw("starting deal", "root", res.Root) - dealProposalCid, err := f.FullNode.ClientStartDeal(ctx, &dp) + // Make a deal + res, err := f.MakeDummyDeal(dealUuid, inPath, root, server.URL+"/"+filepath.Base(inPath), false) require.NoError(t, err) + require.True(t, res.Result.Accepted) + log.Debugw("got response from MarketDummyDeal", "res", spew.Sdump(res)) + dealCid, err := res.DealParams.ClientDealProposal.Proposal.Cid() + require.NoError(t, err) + + time.Sleep(2 * time.Second) - log.Debugw("got deal proposal cid", "cid", dealProposalCid) + log.Debugw("got deal proposal cid", "cid", dealCid.String()) - err = f.WaitDealSealed(ctx, dealProposalCid) + err = f.WaitForDealAddedToSector(res.DealParams.DealUUID) require.NoError(t, err) // Deal is stored and sealed, attempt different retrieval forms retrievalCases := []struct { - name string - selector datamodel.Node - matcherFrom, matcherTo int64 - expectCids []cid.Cid + name string + request trustless.Request + expectCids []cid.Cid }{ { - name: "full file, explore-all", - selector: unixfsnode.UnixFSPathSelectorBuilder("", unixfsnode.ExploreAllRecursivelySelector, false), + name: "full file, explore-all", + request: trustless.Request{ + Root: root, + Scope: trustless.DagScopeAll, + }, expectCids: append([]cid.Cid{root}, leaves...), }, { - name: "slice: 0 to 7MiB", - matcherFrom: 0, - matcherTo: 7 << 20, - expectCids: append([]cid.Cid{root}, leaves...), + name: "slice: 0 to 7MiB", + request: trustless.Request{ + Root: root, + Scope: trustless.DagScopeEntity, + Bytes: &trustless.ByteRange{From: 0, To: ptrInt(7 << 20)}, + }, + expectCids: append([]cid.Cid{root}, leaves...), }, { - name: "slice: 1MiB to 2MiB", - matcherFrom: 1 << 20, - matcherTo: 2 << 20, - expectCids: append([]cid.Cid{root}, leaves[4:9]...), + name: "slice: 1MiB to 2MiB", + request: trustless.Request{ + Root: root, + Scope: trustless.DagScopeEntity, + Bytes: &trustless.ByteRange{From: 1 << 20, To: ptrInt(2 << 20)}, + }, + expectCids: append([]cid.Cid{root}, leaves[4:9]...), }, { - name: "slice: first byte", - matcherFrom: 0, - matcherTo: 1, - expectCids: append([]cid.Cid{root}, leaves[0]), + name: "slice: first byte", + request: trustless.Request{ + Root: root, + Scope: trustless.DagScopeEntity, + Bytes: &trustless.ByteRange{From: 0, To: ptrInt(1)}, + }, + expectCids: append([]cid.Cid{root}, leaves[0]), }, { - name: "slice: last byte", - matcherFrom: 7340031, - matcherTo: 7340032, - expectCids: append([]cid.Cid{root}, leaves[len(leaves)-1]), + name: "slice: last byte", + request: trustless.Request{ + Root: root, + Scope: trustless.DagScopeEntity, + Bytes: &trustless.ByteRange{From: 7340031, To: ptrInt(7340032)}, + }, + expectCids: append([]cid.Cid{root}, leaves[len(leaves)-1]), }, { - name: "slice: last two blocks, negative range, boundary", - matcherFrom: -168000 - 1, - matcherTo: math.MaxInt64, - expectCids: append([]cid.Cid{root}, leaves[len(leaves)-2:]...), + name: "slice: last two blocks, negative range, boundary", + request: trustless.Request{ + Root: root, + Scope: trustless.DagScopeEntity, + Bytes: &trustless.ByteRange{From: -168000 - 1}, + }, + expectCids: append([]cid.Cid{root}, leaves[len(leaves)-2:]...), }, { - name: "slice: last block, negative range, boundary", - matcherFrom: -168000, - matcherTo: math.MaxInt64, - expectCids: append([]cid.Cid{root}, leaves[len(leaves)-1]), + name: "slice: last block, negative range, boundary", + request: trustless.Request{ + Root: root, + Scope: trustless.DagScopeEntity, + Bytes: &trustless.ByteRange{From: -168000}, + }, + expectCids: append([]cid.Cid{root}, leaves[len(leaves)-1]), }, { // In this case we are attempting to traverse beyond the file to a @@ -149,24 +181,26 @@ func TestMarketsV1DealRetrieval(t *testing.T) { // return that. This is not strictly an error case, it's up to the // consumer of this data to verify the path doesn't resolve in the // data they get back. - name: "path beyond file", - selector: unixfsnode.UnixFSPathSelectorBuilder("not/a/path", unixfsnode.ExploreAllRecursivelySelector, false), + name: "path beyond file", + request: trustless.Request{ + Root: root, + Scope: trustless.DagScopeAll, + Path: "not/a/path", + }, expectCids: []cid.Cid{root}, }, } for _, tc := range retrievalCases { t.Run(tc.name, func(t *testing.T) { - selNode := tc.selector - if selNode == nil { - // build a selector from the specified slice matcher range - ssb := builder.NewSelectorSpecBuilder(basicnode.Prototype.Any) - ss := ssb.ExploreInterpretAs("unixfs", ssb.MatcherSubset(tc.matcherFrom, tc.matcherTo)) - selNode = ss.Node() - } + log.Debugw("deal is sealed, starting retrieval", "cid", dealCid.String(), "root", root) - log.Debugw("deal is sealed, starting retrieval", "cid", dealProposalCid, "root", res.Root) - outPath := f.Retrieve(ctx, t, dealProposalCid, res.Root, false, selNode) + outPath := f.Retrieve( + ctx, + t, + tc.request, + false, + ) // Inspect what we got gotCids, err := testutil.CidsInCar(outPath) @@ -185,3 +219,7 @@ func TestMarketsV1DealRetrieval(t *testing.T) { }) } } + +func ptrInt(i int64) *int64 { + return &i +} diff --git a/itests/ipni_publish_test.go b/itests/ipni_publish_test.go index 563c916f9..dfdee16f1 100644 --- a/itests/ipni_publish_test.go +++ b/itests/ipni_publish_test.go @@ -25,7 +25,6 @@ func TestIPNIPublish(t *testing.T) { kit.QuietMiningLogs() framework.SetLogLevel() var opts []framework.FrameworkOpts - opts = append(opts, framework.EnableLegacyDeals(true)) f := framework.NewTestFramework(ctx, t, opts...) err := f.Start() require.NoError(t, err) diff --git a/itests/markets_v1_deal_test.go b/itests/markets_v1_deal_test.go deleted file mode 100644 index 9be1e988d..000000000 --- a/itests/markets_v1_deal_test.go +++ /dev/null @@ -1,55 +0,0 @@ -package itests - -import ( - "context" - "testing" - - "github.com/filecoin-project/boost/itests/framework" - "github.com/filecoin-project/boost/testutil" - lapi "github.com/filecoin-project/lotus/api" - "github.com/filecoin-project/lotus/itests/kit" - "github.com/stretchr/testify/require" -) - -func TestMarketsV1Deal(t *testing.T) { - ctx := context.Background() - log := framework.Log - - kit.QuietMiningLogs() - framework.SetLogLevel() - var opts []framework.FrameworkOpts - opts = append(opts, framework.EnableLegacyDeals(true)) - f := framework.NewTestFramework(ctx, t, opts...) - err := f.Start() - require.NoError(t, err) - defer f.Stop() - - // Create a CAR file - log.Debugw("using tempdir", "dir", f.HomeDir) - rseed := 0 - size := 7 << 20 // 7MiB file - - inPath, err := testutil.CreateRandomFile(f.HomeDir, rseed, size) - require.NoError(t, err) - res, err := f.FullNode.ClientImport(ctx, lapi.FileRef{Path: inPath}) - require.NoError(t, err) - - // Create a new markets v1 deal - dp := f.DefaultMarketsV1DealParams() - dp.Data.Root = res.Root - - log.Debugw("starting deal", "root", res.Root) - dealProposalCid, err := f.FullNode.ClientStartDeal(ctx, &dp) - require.NoError(t, err) - - log.Debugw("got deal proposal cid", "cid", dealProposalCid) - - err = f.WaitDealSealed(ctx, dealProposalCid) - require.NoError(t, err) - - log.Debugw("deal is sealed, starting retrieval", "cid", dealProposalCid, "root", res.Root) - outPath := f.Retrieve(ctx, t, dealProposalCid, res.Root, true, nil) - - log.Debugw("retrieval is done, compare in- and out- files", "in", inPath, "out", outPath) - kit.AssertFilesEqual(t, inPath, outPath) -} diff --git a/itests/markets_v1_offline_deal_test.go b/itests/markets_v1_offline_deal_test.go deleted file mode 100644 index 45c21cbf2..000000000 --- a/itests/markets_v1_offline_deal_test.go +++ /dev/null @@ -1,89 +0,0 @@ -package itests - -import ( - "context" - "fmt" - "path/filepath" - "testing" - "time" - - "github.com/filecoin-project/boost-gfm/storagemarket" - "github.com/filecoin-project/boost/itests/framework" - "github.com/filecoin-project/boost/testutil" - lapi "github.com/filecoin-project/lotus/api" - "github.com/filecoin-project/lotus/itests/kit" - "github.com/stretchr/testify/require" -) - -func TestMarketsV1OfflineDeal(t *testing.T) { - ctx := context.Background() - log := framework.Log - - kit.QuietMiningLogs() - framework.SetLogLevel() - var opts []framework.FrameworkOpts - opts = append(opts, framework.EnableLegacyDeals(true)) - f := framework.NewTestFramework(ctx, t, opts...) - err := f.Start() - require.NoError(t, err) - defer f.Stop() - - // Create a CAR file - log.Debugw("using tempdir", "dir", f.HomeDir) - - rseed := 1 - size := 7 << 20 // 7MiB file - inPath, err := testutil.CreateRandomFile(f.HomeDir, rseed, size) - require.NoError(t, err) - res, err := f.FullNode.ClientImport(ctx, lapi.FileRef{Path: inPath}) - require.NoError(t, err) - - // Get the piece size and commP - rootCid := res.Root - pieceInfo, err := f.FullNode.ClientDealPieceCID(ctx, rootCid) - require.NoError(t, err) - - // Create a new markets v1 deal - dp := f.DefaultMarketsV1DealParams() - dp.Data.Root = res.Root - // Replace with params for manual storage deal (offline deal) - dp.Data.TransferType = storagemarket.TTManual - dp.Data.PieceCid = &pieceInfo.PieceCID - dp.Data.PieceSize = pieceInfo.PieceSize.Unpadded() - - log.Debugw("starting offline deal", "root", res.Root) - dealProposalCid, err := f.FullNode.ClientStartDeal(ctx, &dp) - require.NoError(t, err) - log.Debugw("got deal proposal cid", "cid", dealProposalCid) - - // Wait for the deal to reach StorageDealCheckForAcceptance on the client - cd, err := f.FullNode.ClientGetDealInfo(ctx, *dealProposalCid) - require.NoError(t, err) - require.Eventually(t, func() bool { - cd, _ := f.FullNode.ClientGetDealInfo(ctx, *dealProposalCid) - fmt.Println(storagemarket.DealStates[cd.State]) - return cd.State == storagemarket.StorageDealCheckForAcceptance - }, 60*time.Second, 500*time.Millisecond, "actual deal status is %s", storagemarket.DealStates[cd.State]) - - // Create a CAR file from the raw file - log.Debugw("generate out.car for miner") - carFilePath := filepath.Join(f.HomeDir, "out.car") - err = f.FullNode.ClientGenCar(ctx, lapi.FileRef{Path: inPath}, carFilePath) - require.NoError(t, err) - - // Import the CAR file on the miner - this is the equivalent to - // transferring the file across the wire in a normal (non-offline) deal - log.Debugw("import out.car in boost") - err = f.Boost.MarketImportDealData(ctx, *dealProposalCid, carFilePath) - require.NoError(t, err) - - log.Debugw("wait until offline deal is sealed") - err = f.WaitDealSealed(ctx, dealProposalCid) - require.NoError(t, err) - - log.Debugw("offline deal is sealed, starting retrieval", "cid", dealProposalCid, "root", res.Root) - outPath := f.Retrieve(ctx, t, dealProposalCid, res.Root, true, nil) - - log.Debugw("retrieval of offline deal is done, compare in- and out- files", "in", inPath, "out", outPath) - kit.AssertFilesEqual(t, inPath, outPath) -} diff --git a/itests/multiminer_retrieval_graphsync_test.go b/itests/multiminer_retrieval_graphsync_test.go index 1ef4b52ac..81633074b 100644 --- a/itests/multiminer_retrieval_graphsync_test.go +++ b/itests/multiminer_retrieval_graphsync_test.go @@ -6,6 +6,7 @@ import ( "github.com/filecoin-project/boost/itests/shared" "github.com/filecoin-project/lotus/itests/kit" + trustlessutils "github.com/ipld/go-trustless-utils" ) func TestMultiMinerRetrievalGraphsync(t *testing.T) { @@ -17,7 +18,12 @@ func TestMultiMinerRetrievalGraphsync(t *testing.T) { // - recognize that the deal is for a sector on the first miner // - read the data for the deal from the first miner t.Logf("deal is added to piece, starting retrieval of root %s", rt.RootCid) - outPath := rt.BoostAndMiner2.RetrieveDirect(ctx, t, rt.RootCid, nil, true, nil) + outPath := rt.BoostAndMiner2.Retrieve( + ctx, + t, + trustlessutils.Request{Root: rt.RootCid, Scope: trustlessutils.DagScopeAll}, + true, + ) t.Logf("retrieval is done, compare in- and out- files in: %s, out: %s", rt.SampleFilePath, outPath) kit.AssertFilesEqual(t, rt.SampleFilePath, outPath) diff --git a/itests/shared/multiminer.go b/itests/shared/multiminer.go index 4ce607215..e3980c7e8 100644 --- a/itests/shared/multiminer.go +++ b/itests/shared/multiminer.go @@ -23,6 +23,8 @@ type RetrievalTest struct { SampleFilePath string CarFilepath string RootCid cid.Cid + PieceCid cid.Cid + TempDir string } func RunMultiminerRetrievalTest(t *testing.T, rt func(ctx context.Context, t *testing.T, rt *RetrievalTest)) { @@ -35,7 +37,7 @@ func RunMultiminerRetrievalTest(t *testing.T, rt func(ctx context.Context, t *te // Set up two miners, each with a separate boost instance connected to it ensemble := kit.NewEnsemble(t) var opts []framework.FrameworkOpts - opts = append(opts, framework.EnableLegacyDeals(true), framework.WithEnsemble(ensemble)) + opts = append(opts, framework.WithEnsemble(ensemble)) boostAndMiner1 := framework.NewTestFramework(ctx, t, opts...) boostAndMiner2 := framework.NewTestFramework(ctx, t, opts...) ensemble.Start() @@ -59,7 +61,7 @@ func RunMultiminerRetrievalTest(t *testing.T, rt func(ctx context.Context, t *te // Set up the second boost instance so that it can read sector data // not only from the second miner, but also from the first miner - cfg.Dealmaking.GraphsyncStorageAccessApiInfo = []string{cfg.SectorIndexApiInfo, miner1ApiInfo} + cfg.Retrievals.Graphsync.GraphsyncStorageAccessApiInfo = []string{cfg.SectorIndexApiInfo, miner1ApiInfo} // Set up some other ports so they don't clash cfg.Graphql.Port = 8081 @@ -111,5 +113,7 @@ func RunMultiminerRetrievalTest(t *testing.T, rt func(ctx context.Context, t *te SampleFilePath: randomFilepath, CarFilepath: carFilepath, RootCid: rootCid, + PieceCid: res.DealParams.ClientDealProposal.Proposal.PieceCID, + TempDir: tempdir, }) } diff --git a/lib/legacy/dealmanager.go b/lib/legacy/dealmanager.go index 9a4cd19f9..ecdbda048 100644 --- a/lib/legacy/dealmanager.go +++ b/lib/legacy/dealmanager.go @@ -3,18 +3,34 @@ package legacy import ( "context" "errors" - gfm_storagemarket "github.com/filecoin-project/boost-gfm/storagemarket" + "sort" + "sync" + "time" + + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" + "github.com/filecoin-project/go-statemachine/fsm" "github.com/ipfs/go-cid" "github.com/ipfs/go-datastore" logging "github.com/ipfs/go-log/v2" - "sync" - "time" ) +//go:generate go run github.com/golang/mock/mockgen -destination=mocks/legacy_manager_mock.go . LegacyDealManager + +type LegacyDealManager interface { + Run(ctx context.Context) + DealCount(ctx context.Context) (int, error) + ByPieceCid(ctx context.Context, pieceCid cid.Cid) ([]legacytypes.MinerDeal, error) + ByPayloadCid(ctx context.Context, payloadCid cid.Cid) ([]legacytypes.MinerDeal, error) + ByPublishCid(ctx context.Context, publishCid cid.Cid) ([]legacytypes.MinerDeal, error) + ListDeals() ([]legacytypes.MinerDeal, error) + ByPropCid(propCid cid.Cid) (legacytypes.MinerDeal, error) + ListLocalDealsPage(startPropCid *cid.Cid, offset int, limit int) ([]legacytypes.MinerDeal, error) +} + var log = logging.Logger("legacydeals") -type LegacyDealsManager struct { - legacyProv gfm_storagemarket.StorageProvider +type legacyDealsManager struct { + legacyFSM fsm.Group startedOnce sync.Once started chan struct{} @@ -26,9 +42,9 @@ type LegacyDealsManager struct { publishCidIdx map[cid.Cid][]cid.Cid } -func NewLegacyDealsManager(legacyProv gfm_storagemarket.StorageProvider) *LegacyDealsManager { - return &LegacyDealsManager{ - legacyProv: legacyProv, +func NewLegacyDealsManager(legacyFSM fsm.Group) *legacyDealsManager { + return &legacyDealsManager{ + legacyFSM: legacyFSM, started: make(chan struct{}), pieceCidIdx: make(map[cid.Cid][]cid.Cid), payloadCidIdx: make(map[cid.Cid][]cid.Cid), @@ -36,7 +52,7 @@ func NewLegacyDealsManager(legacyProv gfm_storagemarket.StorageProvider) *Legacy } } -func (m *LegacyDealsManager) Run(ctx context.Context) { +func (m *legacyDealsManager) Run(ctx context.Context) { refresh := func() { err := m.refresh() if err != nil { @@ -60,10 +76,10 @@ func (m *LegacyDealsManager) Run(ctx context.Context) { } } -func (m *LegacyDealsManager) refresh() error { +func (m *legacyDealsManager) refresh() error { start := time.Now() log.Infow("refreshing legacy deals list") - dls, err := m.legacyProv.ListLocalDeals() + dls, err := m.ListDeals() if err != nil { return err } @@ -101,7 +117,7 @@ func (m *LegacyDealsManager) refresh() error { return nil } -func (m *LegacyDealsManager) waitStarted(ctx context.Context) error { +func (m *legacyDealsManager) waitStarted(ctx context.Context) error { select { case <-ctx.Done(): return ctx.Err() @@ -110,7 +126,7 @@ func (m *LegacyDealsManager) waitStarted(ctx context.Context) error { } } -func (m *LegacyDealsManager) DealCount(ctx context.Context) (int, error) { +func (m *legacyDealsManager) DealCount(ctx context.Context) (int, error) { if err := m.waitStarted(ctx); err != nil { return 0, err } @@ -121,7 +137,7 @@ func (m *LegacyDealsManager) DealCount(ctx context.Context) (int, error) { return m.dealCount, nil } -func (m *LegacyDealsManager) ByPieceCid(ctx context.Context, pieceCid cid.Cid) ([]gfm_storagemarket.MinerDeal, error) { +func (m *legacyDealsManager) ByPieceCid(ctx context.Context, pieceCid cid.Cid) ([]legacytypes.MinerDeal, error) { if err := m.waitStarted(ctx); err != nil { return nil, err } @@ -137,7 +153,7 @@ func (m *LegacyDealsManager) ByPieceCid(ctx context.Context, pieceCid cid.Cid) ( return m.byPropCids(propCids) } -func (m *LegacyDealsManager) ByPayloadCid(ctx context.Context, payloadCid cid.Cid) ([]gfm_storagemarket.MinerDeal, error) { +func (m *legacyDealsManager) ByPayloadCid(ctx context.Context, payloadCid cid.Cid) ([]legacytypes.MinerDeal, error) { if err := m.waitStarted(ctx); err != nil { return nil, err } @@ -153,7 +169,7 @@ func (m *LegacyDealsManager) ByPayloadCid(ctx context.Context, payloadCid cid.Ci return m.byPropCids(propCids) } -func (m *LegacyDealsManager) ByPublishCid(ctx context.Context, publishCid cid.Cid) ([]gfm_storagemarket.MinerDeal, error) { +func (m *legacyDealsManager) ByPublishCid(ctx context.Context, publishCid cid.Cid) ([]legacytypes.MinerDeal, error) { if err := m.waitStarted(ctx); err != nil { return nil, err } @@ -170,12 +186,13 @@ func (m *LegacyDealsManager) ByPublishCid(ctx context.Context, publishCid cid.Ci } // Get deals by deal signed proposal cid -func (m *LegacyDealsManager) byPropCids(propCids []cid.Cid) ([]gfm_storagemarket.MinerDeal, error) { - dls := make([]gfm_storagemarket.MinerDeal, 0, len(propCids)) +func (m *legacyDealsManager) byPropCids(propCids []cid.Cid) ([]legacytypes.MinerDeal, error) { + dls := make([]legacytypes.MinerDeal, 0, len(propCids)) for _, propCid := range propCids { - dl, err := m.legacyProv.GetLocalDeal(propCid) + var d legacytypes.MinerDeal + err := m.legacyFSM.Get(propCid).Get(&d) if err == nil { - dls = append(dls, dl) + dls = append(dls, d) continue } @@ -188,3 +205,62 @@ func (m *LegacyDealsManager) byPropCids(propCids []cid.Cid) ([]gfm_storagemarket return dls, nil } + +func (m *legacyDealsManager) ListDeals() ([]legacytypes.MinerDeal, error) { + var list []legacytypes.MinerDeal + if err := m.legacyFSM.List(&list); err != nil { + return nil, err + } + return list, nil +} + +// Get deal by deal signed proposal cid +func (m *legacyDealsManager) ByPropCid(propCid cid.Cid) (legacytypes.MinerDeal, error) { + var d legacytypes.MinerDeal + err := m.legacyFSM.Get(propCid).Get(&d) + if err != nil { + return legacytypes.MinerDeal{}, err + } + return d, nil +} + +func (m *legacyDealsManager) ListLocalDealsPage(startPropCid *cid.Cid, offset int, limit int) ([]legacytypes.MinerDeal, error) { + if limit == 0 { + return []legacytypes.MinerDeal{}, nil + } + + // Get all deals + var deals []legacytypes.MinerDeal + if err := m.legacyFSM.List(&deals); err != nil { + return nil, err + } + + // Sort by creation time descending + sort.Slice(deals, func(i, j int) bool { + return deals[i].CreationTime.Time().After(deals[j].CreationTime.Time()) + }) + + // Iterate through deals until we reach the target signed proposal cid, + // find the offset from there, then add deals from that point up to limit + page := make([]legacytypes.MinerDeal, 0, limit) + startIndex := -1 + if startPropCid == nil { + startIndex = 0 + } + for i, dl := range deals { + // Find the deal with a proposal cid matching startPropCid + if startPropCid != nil && dl.ProposalCid == *startPropCid { + // Start adding deals from offset after the first matching deal + startIndex = i + offset + } + + if startIndex >= 0 && i >= startIndex { + page = append(page, dl) + } + if len(page) == limit { + return page, nil + } + } + + return page, nil +} diff --git a/lib/legacy/mocks/legacy_manager_mock.go b/lib/legacy/mocks/legacy_manager_mock.go new file mode 100644 index 000000000..d61c5adbd --- /dev/null +++ b/lib/legacy/mocks/legacy_manager_mock.go @@ -0,0 +1,154 @@ +// Code generated by MockGen. DO NOT EDIT. +// Source: github.com/filecoin-project/boost/lib/legacy (interfaces: LegacyDealManager) + +// Package mock_legacy is a generated GoMock package. +package mock_legacy + +import ( + context "context" + reflect "reflect" + + legacytypes "github.com/filecoin-project/boost/storagemarket/types/legacytypes" + gomock "github.com/golang/mock/gomock" + cid "github.com/ipfs/go-cid" +) + +// MockLegacyDealManager is a mock of LegacyDealManager interface. +type MockLegacyDealManager struct { + ctrl *gomock.Controller + recorder *MockLegacyDealManagerMockRecorder +} + +// MockLegacyDealManagerMockRecorder is the mock recorder for MockLegacyDealManager. +type MockLegacyDealManagerMockRecorder struct { + mock *MockLegacyDealManager +} + +// NewMockLegacyDealManager creates a new mock instance. +func NewMockLegacyDealManager(ctrl *gomock.Controller) *MockLegacyDealManager { + mock := &MockLegacyDealManager{ctrl: ctrl} + mock.recorder = &MockLegacyDealManagerMockRecorder{mock} + return mock +} + +// EXPECT returns an object that allows the caller to indicate expected use. +func (m *MockLegacyDealManager) EXPECT() *MockLegacyDealManagerMockRecorder { + return m.recorder +} + +// ByPayloadCid mocks base method. +func (m *MockLegacyDealManager) ByPayloadCid(arg0 context.Context, arg1 cid.Cid) ([]legacytypes.MinerDeal, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "ByPayloadCid", arg0, arg1) + ret0, _ := ret[0].([]legacytypes.MinerDeal) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// ByPayloadCid indicates an expected call of ByPayloadCid. +func (mr *MockLegacyDealManagerMockRecorder) ByPayloadCid(arg0, arg1 interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "ByPayloadCid", reflect.TypeOf((*MockLegacyDealManager)(nil).ByPayloadCid), arg0, arg1) +} + +// ByPieceCid mocks base method. +func (m *MockLegacyDealManager) ByPieceCid(arg0 context.Context, arg1 cid.Cid) ([]legacytypes.MinerDeal, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "ByPieceCid", arg0, arg1) + ret0, _ := ret[0].([]legacytypes.MinerDeal) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// ByPieceCid indicates an expected call of ByPieceCid. +func (mr *MockLegacyDealManagerMockRecorder) ByPieceCid(arg0, arg1 interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "ByPieceCid", reflect.TypeOf((*MockLegacyDealManager)(nil).ByPieceCid), arg0, arg1) +} + +// ByPropCid mocks base method. +func (m *MockLegacyDealManager) ByPropCid(arg0 cid.Cid) (legacytypes.MinerDeal, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "ByPropCid", arg0) + ret0, _ := ret[0].(legacytypes.MinerDeal) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// ByPropCid indicates an expected call of ByPropCid. +func (mr *MockLegacyDealManagerMockRecorder) ByPropCid(arg0 interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "ByPropCid", reflect.TypeOf((*MockLegacyDealManager)(nil).ByPropCid), arg0) +} + +// ByPublishCid mocks base method. +func (m *MockLegacyDealManager) ByPublishCid(arg0 context.Context, arg1 cid.Cid) ([]legacytypes.MinerDeal, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "ByPublishCid", arg0, arg1) + ret0, _ := ret[0].([]legacytypes.MinerDeal) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// ByPublishCid indicates an expected call of ByPublishCid. +func (mr *MockLegacyDealManagerMockRecorder) ByPublishCid(arg0, arg1 interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "ByPublishCid", reflect.TypeOf((*MockLegacyDealManager)(nil).ByPublishCid), arg0, arg1) +} + +// DealCount mocks base method. +func (m *MockLegacyDealManager) DealCount(arg0 context.Context) (int, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "DealCount", arg0) + ret0, _ := ret[0].(int) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// DealCount indicates an expected call of DealCount. +func (mr *MockLegacyDealManagerMockRecorder) DealCount(arg0 interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "DealCount", reflect.TypeOf((*MockLegacyDealManager)(nil).DealCount), arg0) +} + +// ListDeals mocks base method. +func (m *MockLegacyDealManager) ListDeals() ([]legacytypes.MinerDeal, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "ListDeals") + ret0, _ := ret[0].([]legacytypes.MinerDeal) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// ListDeals indicates an expected call of ListDeals. +func (mr *MockLegacyDealManagerMockRecorder) ListDeals() *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "ListDeals", reflect.TypeOf((*MockLegacyDealManager)(nil).ListDeals)) +} + +// ListLocalDealsPage mocks base method. +func (m *MockLegacyDealManager) ListLocalDealsPage(arg0 *cid.Cid, arg1, arg2 int) ([]legacytypes.MinerDeal, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "ListLocalDealsPage", arg0, arg1, arg2) + ret0, _ := ret[0].([]legacytypes.MinerDeal) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// ListLocalDealsPage indicates an expected call of ListLocalDealsPage. +func (mr *MockLegacyDealManagerMockRecorder) ListLocalDealsPage(arg0, arg1, arg2 interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "ListLocalDealsPage", reflect.TypeOf((*MockLegacyDealManager)(nil).ListLocalDealsPage), arg0, arg1, arg2) +} + +// Run mocks base method. +func (m *MockLegacyDealManager) Run(arg0 context.Context) { + m.ctrl.T.Helper() + m.ctrl.Call(m, "Run", arg0) +} + +// Run indicates an expected call of Run. +func (mr *MockLegacyDealManagerMockRecorder) Run(arg0 interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Run", reflect.TypeOf((*MockLegacyDealManager)(nil).Run), arg0) +} diff --git a/markets/journal.go b/markets/journal.go deleted file mode 100644 index 8de216f2a..000000000 --- a/markets/journal.go +++ /dev/null @@ -1,76 +0,0 @@ -package markets - -import ( - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/boost-gfm/storagemarket" - - "github.com/filecoin-project/lotus/journal" -) - -type StorageClientEvt struct { - Event string - Deal storagemarket.ClientDeal -} - -type StorageProviderEvt struct { - Event string - Deal storagemarket.MinerDeal -} - -type RetrievalClientEvt struct { - Event string - Deal retrievalmarket.ClientDealState -} - -type RetrievalProviderEvt struct { - Event string - Deal retrievalmarket.ProviderDealState -} - -// StorageClientJournaler records journal events from the storage client. -func StorageClientJournaler(j journal.Journal, evtType journal.EventType) func(event storagemarket.ClientEvent, deal storagemarket.ClientDeal) { - return func(event storagemarket.ClientEvent, deal storagemarket.ClientDeal) { - j.RecordEvent(evtType, func() interface{} { - return StorageClientEvt{ - Event: storagemarket.ClientEvents[event], - Deal: deal, - } - }) - } -} - -// StorageProviderJournaler records journal events from the storage provider. -func StorageProviderJournaler(j journal.Journal, evtType journal.EventType) func(event storagemarket.ProviderEvent, deal storagemarket.MinerDeal) { - return func(event storagemarket.ProviderEvent, deal storagemarket.MinerDeal) { - j.RecordEvent(evtType, func() interface{} { - return StorageProviderEvt{ - Event: storagemarket.ProviderEvents[event], - Deal: deal, - } - }) - } -} - -// RetrievalClientJournaler records journal events from the retrieval client. -func RetrievalClientJournaler(j journal.Journal, evtType journal.EventType) func(event retrievalmarket.ClientEvent, deal retrievalmarket.ClientDealState) { - return func(event retrievalmarket.ClientEvent, deal retrievalmarket.ClientDealState) { - j.RecordEvent(evtType, func() interface{} { - return RetrievalClientEvt{ - Event: retrievalmarket.ClientEvents[event], - Deal: deal, - } - }) - } -} - -// RetrievalProviderJournaler records journal events from the retrieval provider. -func RetrievalProviderJournaler(j journal.Journal, evtType journal.EventType) func(event retrievalmarket.ProviderEvent, deal retrievalmarket.ProviderDealState) { - return func(event retrievalmarket.ProviderEvent, deal retrievalmarket.ProviderDealState) { - j.RecordEvent(evtType, func() interface{} { - return RetrievalProviderEvt{ - Event: retrievalmarket.ProviderEvents[event], - Deal: deal, - } - }) - } -} diff --git a/markets/loggers/loggers.go b/markets/loggers/loggers.go index 8cdf13402..e765d65f7 100644 --- a/markets/loggers/loggers.go +++ b/markets/loggers/loggers.go @@ -1,49 +1,38 @@ package marketevents import ( + datatransfer2 "github.com/filecoin-project/boost/datatransfer" logging "github.com/ipfs/go-log/v2" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/boost-gfm/storagemarket" - datatransfer "github.com/filecoin-project/go-data-transfer" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" "github.com/filecoin-project/go-state-types/abi" ) var log = logging.Logger("markets") -// StorageClientLogger logs events from the storage client -func StorageClientLogger(event storagemarket.ClientEvent, deal storagemarket.ClientDeal) { - log.Infow("storage client event", "name", storagemarket.ClientEvents[event], "proposal CID", deal.ProposalCid, "state", storagemarket.DealStates[deal.State], "message", deal.Message) -} - -// StorageProviderLogger logs events from the storage provider -func StorageProviderLogger(event storagemarket.ProviderEvent, deal storagemarket.MinerDeal) { - log.Infow("storage provider event", "name", storagemarket.ProviderEvents[event], "proposal CID", deal.ProposalCid, "state", storagemarket.DealStates[deal.State], "message", deal.Message) -} - // RetrievalClientLogger logs events from the retrieval client -func RetrievalClientLogger(event retrievalmarket.ClientEvent, deal retrievalmarket.ClientDealState) { +func RetrievalClientLogger(event legacyretrievaltypes.ClientEvent, deal legacyretrievaltypes.ClientDealState) { method := log.Infow - if event == retrievalmarket.ClientEventBlocksReceived { + if event == legacyretrievaltypes.ClientEventBlocksReceived { method = log.Debugw } - method("retrieval client event", "name", retrievalmarket.ClientEvents[event], "deal ID", deal.ID, "state", retrievalmarket.DealStatuses[deal.Status], "message", deal.Message) + method("retrieval client event", "name", legacyretrievaltypes.ClientEvents[event], "deal ID", deal.ID, "state", legacyretrievaltypes.DealStatuses[deal.Status], "message", deal.Message) } // RetrievalProviderLogger logs events from the retrieval provider -func RetrievalProviderLogger(event retrievalmarket.ProviderEvent, deal retrievalmarket.ProviderDealState) { +func RetrievalProviderLogger(event legacyretrievaltypes.ProviderEvent, deal legacyretrievaltypes.ProviderDealState) { method := log.Infow - if event == retrievalmarket.ProviderEventBlockSent { + if event == legacyretrievaltypes.ProviderEventBlockSent { method = log.Debugw } - method("retrieval provider event", "name", retrievalmarket.ProviderEvents[event], "deal ID", deal.ID, "receiver", deal.Receiver, "state", retrievalmarket.DealStatuses[deal.Status], "message", deal.Message) + method("retrieval provider event", "name", legacyretrievaltypes.ProviderEvents[event], "deal ID", deal.ID, "receiver", deal.Receiver, "state", legacyretrievaltypes.DealStatuses[deal.Status], "message", deal.Message) } // DataTransferLogger logs events from the data transfer module -func DataTransferLogger(event datatransfer.Event, state datatransfer.ChannelState) { +func DataTransferLogger(event datatransfer2.Event, state datatransfer2.ChannelState) { log.Debugw("data transfer event", - "name", datatransfer.Events[event.Code], - "status", datatransfer.Statuses[state.Status()], + "name", datatransfer2.Events[event.Code], + "status", datatransfer2.Statuses[state.Status()], "transfer ID", state.TransferID(), "channel ID", state.ChannelID(), "sent", state.Sent(), @@ -68,8 +57,8 @@ func ReadyLogger(module string) func(error) { } type RetrievalEvent struct { - Event retrievalmarket.ClientEvent - Status retrievalmarket.DealStatus + Event legacyretrievaltypes.ClientEvent + Status legacyretrievaltypes.DealStatus BytesReceived uint64 FundsSpent abi.TokenAmount Err string diff --git a/markets/piecestore/impl/piecestore.go b/markets/piecestore/impl/piecestore.go new file mode 100644 index 000000000..12bc235d7 --- /dev/null +++ b/markets/piecestore/impl/piecestore.go @@ -0,0 +1,214 @@ +package piecestoreimpl + +import ( + "context" + "errors" + "fmt" + + "github.com/filecoin-project/boost/markets/piecestore" + "github.com/filecoin-project/boost/markets/piecestore/migrations" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" + versioning "github.com/filecoin-project/go-ds-versioning/pkg" + versioned "github.com/filecoin-project/go-ds-versioning/pkg/statestore" + "github.com/hannahhoward/go-pubsub" + "github.com/ipfs/go-cid" + "github.com/ipfs/go-datastore" + "github.com/ipfs/go-datastore/namespace" + logging "github.com/ipfs/go-log/v2" + + "github.com/filecoin-project/boost/markets/shared" +) + +var log = logging.Logger("piecestore") + +// DSPiecePrefix is the name space for storing piece infos +var DSPiecePrefix = "/pieces" + +// DSCIDPrefix is the name space for storing CID infos +var DSCIDPrefix = "/cid-infos" + +// NewPieceStore returns a new piecestore based on the given datastore +func NewPieceStore(ds datastore.Batching) (*pieceStore, error) { + pieceInfoMigrations, err := migrations.PieceInfoMigrations.Build() + if err != nil { + return nil, err + } + pieces, migratePieces := versioned.NewVersionedStateStore(namespace.Wrap(ds, datastore.NewKey(DSPiecePrefix)), pieceInfoMigrations, versioning.VersionKey("1")) + cidInfoMigrations, err := migrations.CIDInfoMigrations.Build() + if err != nil { + return nil, err + } + cidInfos, migrateCidInfos := versioned.NewVersionedStateStore(namespace.Wrap(ds, datastore.NewKey(DSCIDPrefix)), cidInfoMigrations, versioning.VersionKey("1")) + return &pieceStore{ + readySub: pubsub.New(shared.ReadyDispatcher), + pieces: pieces, + migratePieces: migratePieces, + cidInfos: cidInfos, + migrateCidInfos: migrateCidInfos, + }, nil +} + +type pieceStore struct { + readySub *pubsub.PubSub + migratePieces func(ctx context.Context) error + pieces versioned.StateStore + migrateCidInfos func(ctx context.Context) error + cidInfos versioned.StateStore +} + +func (ps *pieceStore) Start(ctx context.Context) error { + go func() { + var err error + defer func() { + err = ps.readySub.Publish(err) + if err != nil { + log.Warnf("Publish piecestore migration ready event: %s", err.Error()) + } + }() + err = ps.migratePieces(ctx) + if err != nil { + log.Errorf("Migrating pieceInfos: %s", err.Error()) + return + } + err = ps.migrateCidInfos(ctx) + if err != nil { + log.Errorf("Migrating cidInfos: %s", err.Error()) + } + }() + return nil +} + +func (ps *pieceStore) OnReady(ready shared.ReadyFunc) { + ps.readySub.Subscribe(ready) +} + +// Store `dealInfo` in the PieceStore with key `pieceCID`. +func (ps *pieceStore) AddDealForPiece(pieceCID cid.Cid, _ cid.Cid, dealInfo piecestore.DealInfo) error { + return ps.mutatePieceInfo(pieceCID, func(pi *piecestore.PieceInfo) error { + for _, di := range pi.Deals { + if di == dealInfo { + return nil + } + } + pi.Deals = append(pi.Deals, dealInfo) + return nil + }) +} + +// Store the map of blockLocations in the PieceStore's CIDInfo store, with key `pieceCID` +func (ps *pieceStore) AddPieceBlockLocations(pieceCID cid.Cid, blockLocations map[cid.Cid]piecestore.BlockLocation) error { + for c, blockLocation := range blockLocations { + err := ps.mutateCIDInfo(c, func(ci *piecestore.CIDInfo) error { + for _, pbl := range ci.PieceBlockLocations { + if pbl.PieceCID.Equals(pieceCID) && pbl.BlockLocation == blockLocation { + return nil + } + } + ci.PieceBlockLocations = append(ci.PieceBlockLocations, piecestore.PieceBlockLocation{BlockLocation: blockLocation, PieceCID: pieceCID}) + return nil + }) + if err != nil { + return err + } + } + return nil +} + +func (ps *pieceStore) ListPieceInfoKeys() ([]cid.Cid, error) { + var pis []piecestore.PieceInfo + if err := ps.pieces.List(&pis); err != nil { + return nil, err + } + + out := make([]cid.Cid, 0, len(pis)) + for _, pi := range pis { + out = append(out, pi.PieceCID) + } + + return out, nil +} + +func (ps *pieceStore) ListCidInfoKeys() ([]cid.Cid, error) { + var cis []piecestore.CIDInfo + if err := ps.cidInfos.List(&cis); err != nil { + return nil, err + } + + out := make([]cid.Cid, 0, len(cis)) + for _, ci := range cis { + out = append(out, ci.CID) + } + + return out, nil +} + +// Retrieve the PieceInfo associated with `pieceCID` from the piece info store. +func (ps *pieceStore) GetPieceInfo(pieceCID cid.Cid) (piecestore.PieceInfo, error) { + var out piecestore.PieceInfo + if err := ps.pieces.Get(pieceCID).Get(&out); err != nil { + if errors.Is(err, datastore.ErrNotFound) { + return piecestore.PieceInfo{}, fmt.Errorf("piece with CID %s: %w", pieceCID, legacyretrievaltypes.ErrNotFound) + } + return piecestore.PieceInfo{}, err + } + return out, nil +} + +// Retrieve the CIDInfo associated with `pieceCID` from the CID info store. +func (ps *pieceStore) GetCIDInfo(payloadCID cid.Cid) (piecestore.CIDInfo, error) { + var out piecestore.CIDInfo + if err := ps.cidInfos.Get(payloadCID).Get(&out); err != nil { + if errors.Is(err, datastore.ErrNotFound) { + return piecestore.CIDInfo{}, fmt.Errorf("payload CID %s: %w", payloadCID, legacyretrievaltypes.ErrNotFound) + } + return piecestore.CIDInfo{}, err + } + return out, nil +} + +func (ps *pieceStore) ensurePieceInfo(pieceCID cid.Cid) error { + has, err := ps.pieces.Has(pieceCID) + + if err != nil { + return err + } + if has { + return nil + } + + pieceInfo := piecestore.PieceInfo{PieceCID: pieceCID} + return ps.pieces.Begin(pieceCID, &pieceInfo) +} + +func (ps *pieceStore) ensureCIDInfo(c cid.Cid) error { + has, err := ps.cidInfos.Has(c) + + if err != nil { + return err + } + + if has { + return nil + } + + cidInfo := piecestore.CIDInfo{CID: c} + return ps.cidInfos.Begin(c, &cidInfo) +} + +func (ps *pieceStore) mutatePieceInfo(pieceCID cid.Cid, mutator interface{}) error { + err := ps.ensurePieceInfo(pieceCID) + if err != nil { + return err + } + + return ps.pieces.Get(pieceCID).Mutate(mutator) +} + +func (ps *pieceStore) mutateCIDInfo(c cid.Cid, mutator interface{}) error { + err := ps.ensureCIDInfo(c) + if err != nil { + return err + } + + return ps.cidInfos.Get(c).Mutate(mutator) +} diff --git a/markets/piecestore/migrations/migrations.go b/markets/piecestore/migrations/migrations.go new file mode 100644 index 000000000..b3ec99f45 --- /dev/null +++ b/markets/piecestore/migrations/migrations.go @@ -0,0 +1,90 @@ +package migrations + +import ( + "github.com/filecoin-project/boost/markets/piecestore" + "github.com/ipfs/go-cid" + + versioning "github.com/filecoin-project/go-ds-versioning/pkg" + "github.com/filecoin-project/go-ds-versioning/pkg/versioned" + "github.com/filecoin-project/go-state-types/abi" +) + +//go:generate cbor-gen-for PieceInfo0 DealInfo0 BlockLocation0 PieceBlockLocation0 CIDInfo0 + +// DealInfo0 is version 0 of DealInfo +type DealInfo0 struct { + DealID abi.DealID + SectorID abi.SectorNumber + Offset abi.PaddedPieceSize + Length abi.PaddedPieceSize +} + +// BlockLocation0 is version 0 of BlockLocation +type BlockLocation0 struct { + RelOffset uint64 + BlockSize uint64 +} + +// PieceBlockLocation0 is version 0 of PieceBlockLocation +// is inside of +type PieceBlockLocation0 struct { + BlockLocation0 + PieceCID cid.Cid +} + +// CIDInfo0 is version 0 of CIDInfo +type CIDInfo0 struct { + CID cid.Cid + PieceBlockLocations []PieceBlockLocation0 +} + +// PieceInfo0 is version 0 of PieceInfo +type PieceInfo0 struct { + PieceCID cid.Cid + Deals []DealInfo0 +} + +// MigratePieceInfo0To1 migrates a tuple encoded piece info to a map encoded piece info +func MigratePieceInfo0To1(oldPi *PieceInfo0) (*piecestore.PieceInfo, error) { + deals := make([]piecestore.DealInfo, len(oldPi.Deals)) + for i, oldDi := range oldPi.Deals { + deals[i] = piecestore.DealInfo{ + DealID: oldDi.DealID, + SectorID: oldDi.SectorID, + Offset: oldDi.Offset, + Length: oldDi.Length, + } + } + return &piecestore.PieceInfo{ + PieceCID: oldPi.PieceCID, + Deals: deals, + }, nil +} + +// MigrateCidInfo0To1 migrates a tuple encoded cid info to a map encoded cid info +func MigrateCidInfo0To1(oldCi *CIDInfo0) (*piecestore.CIDInfo, error) { + pieceBlockLocations := make([]piecestore.PieceBlockLocation, len(oldCi.PieceBlockLocations)) + for i, oldPbl := range oldCi.PieceBlockLocations { + pieceBlockLocations[i] = piecestore.PieceBlockLocation{ + BlockLocation: piecestore.BlockLocation{ + RelOffset: oldPbl.RelOffset, + BlockSize: oldPbl.BlockSize, + }, + PieceCID: oldPbl.PieceCID, + } + } + return &piecestore.CIDInfo{ + CID: oldCi.CID, + PieceBlockLocations: pieceBlockLocations, + }, nil +} + +// PieceInfoMigrations is the list of migrations for migrating PieceInfos +var PieceInfoMigrations = versioned.BuilderList{ + versioned.NewVersionedBuilder(MigratePieceInfo0To1, versioning.VersionKey("1")), +} + +// CIDInfoMigrations is the list of migrations for migrating CIDInfos +var CIDInfoMigrations = versioned.BuilderList{ + versioned.NewVersionedBuilder(MigrateCidInfo0To1, versioning.VersionKey("1")), +} diff --git a/markets/piecestore/migrations/migrations_cbor_gen.go b/markets/piecestore/migrations/migrations_cbor_gen.go new file mode 100644 index 000000000..ff2513f6f --- /dev/null +++ b/markets/piecestore/migrations/migrations_cbor_gen.go @@ -0,0 +1,527 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package migrations + +import ( + "fmt" + "io" + "math" + "sort" + + abi "github.com/filecoin-project/go-state-types/abi" + cid "github.com/ipfs/go-cid" + cbg "github.com/whyrusleeping/cbor-gen" + xerrors "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +var lengthBufPieceInfo0 = []byte{130} + +func (t *PieceInfo0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufPieceInfo0); err != nil { + return err + } + + // t.PieceCID (cid.Cid) (struct) + + if err := cbg.WriteCid(cw, t.PieceCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PieceCID: %w", err) + } + + // t.Deals ([]migrations.DealInfo0) (slice) + if len(t.Deals) > cbg.MaxLength { + return xerrors.Errorf("Slice value in field t.Deals was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajArray, uint64(len(t.Deals))); err != nil { + return err + } + for _, v := range t.Deals { + if err := v.MarshalCBOR(cw); err != nil { + return err + } + + } + return nil +} + +func (t *PieceInfo0) UnmarshalCBOR(r io.Reader) (err error) { + *t = PieceInfo0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 2 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.PieceCID (cid.Cid) (struct) + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PieceCID: %w", err) + } + + t.PieceCID = c + + } + // t.Deals ([]migrations.DealInfo0) (slice) + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + + if extra > cbg.MaxLength { + return fmt.Errorf("t.Deals: array too large (%d)", extra) + } + + if maj != cbg.MajArray { + return fmt.Errorf("expected cbor array") + } + + if extra > 0 { + t.Deals = make([]DealInfo0, extra) + } + + for i := 0; i < int(extra); i++ { + { + var maj byte + var extra uint64 + var err error + _ = maj + _ = extra + _ = err + + { + + if err := t.Deals[i].UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Deals[i]: %w", err) + } + + } + + } + } + return nil +} + +var lengthBufDealInfo0 = []byte{132} + +func (t *DealInfo0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufDealInfo0); err != nil { + return err + } + + // t.DealID (abi.DealID) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.DealID)); err != nil { + return err + } + + // t.SectorID (abi.SectorNumber) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.SectorID)); err != nil { + return err + } + + // t.Offset (abi.PaddedPieceSize) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Offset)); err != nil { + return err + } + + // t.Length (abi.PaddedPieceSize) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Length)); err != nil { + return err + } + + return nil +} + +func (t *DealInfo0) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealInfo0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 4 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.DealID (abi.DealID) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.DealID = abi.DealID(extra) + + } + // t.SectorID (abi.SectorNumber) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.SectorID = abi.SectorNumber(extra) + + } + // t.Offset (abi.PaddedPieceSize) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Offset = abi.PaddedPieceSize(extra) + + } + // t.Length (abi.PaddedPieceSize) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Length = abi.PaddedPieceSize(extra) + + } + return nil +} + +var lengthBufBlockLocation0 = []byte{130} + +func (t *BlockLocation0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufBlockLocation0); err != nil { + return err + } + + // t.RelOffset (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.RelOffset)); err != nil { + return err + } + + // t.BlockSize (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.BlockSize)); err != nil { + return err + } + + return nil +} + +func (t *BlockLocation0) UnmarshalCBOR(r io.Reader) (err error) { + *t = BlockLocation0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 2 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.RelOffset (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.RelOffset = uint64(extra) + + } + // t.BlockSize (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.BlockSize = uint64(extra) + + } + return nil +} + +var lengthBufPieceBlockLocation0 = []byte{130} + +func (t *PieceBlockLocation0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufPieceBlockLocation0); err != nil { + return err + } + + // t.BlockLocation0 (migrations.BlockLocation0) (struct) + if err := t.BlockLocation0.MarshalCBOR(cw); err != nil { + return err + } + + // t.PieceCID (cid.Cid) (struct) + + if err := cbg.WriteCid(cw, t.PieceCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PieceCID: %w", err) + } + + return nil +} + +func (t *PieceBlockLocation0) UnmarshalCBOR(r io.Reader) (err error) { + *t = PieceBlockLocation0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 2 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.BlockLocation0 (migrations.BlockLocation0) (struct) + + { + + if err := t.BlockLocation0.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.BlockLocation0: %w", err) + } + + } + // t.PieceCID (cid.Cid) (struct) + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PieceCID: %w", err) + } + + t.PieceCID = c + + } + return nil +} + +var lengthBufCIDInfo0 = []byte{130} + +func (t *CIDInfo0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufCIDInfo0); err != nil { + return err + } + + // t.CID (cid.Cid) (struct) + + if err := cbg.WriteCid(cw, t.CID); err != nil { + return xerrors.Errorf("failed to write cid field t.CID: %w", err) + } + + // t.PieceBlockLocations ([]migrations.PieceBlockLocation0) (slice) + if len(t.PieceBlockLocations) > cbg.MaxLength { + return xerrors.Errorf("Slice value in field t.PieceBlockLocations was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajArray, uint64(len(t.PieceBlockLocations))); err != nil { + return err + } + for _, v := range t.PieceBlockLocations { + if err := v.MarshalCBOR(cw); err != nil { + return err + } + + } + return nil +} + +func (t *CIDInfo0) UnmarshalCBOR(r io.Reader) (err error) { + *t = CIDInfo0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 2 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.CID (cid.Cid) (struct) + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.CID: %w", err) + } + + t.CID = c + + } + // t.PieceBlockLocations ([]migrations.PieceBlockLocation0) (slice) + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + + if extra > cbg.MaxLength { + return fmt.Errorf("t.PieceBlockLocations: array too large (%d)", extra) + } + + if maj != cbg.MajArray { + return fmt.Errorf("expected cbor array") + } + + if extra > 0 { + t.PieceBlockLocations = make([]PieceBlockLocation0, extra) + } + + for i := 0; i < int(extra); i++ { + { + var maj byte + var extra uint64 + var err error + _ = maj + _ = extra + _ = err + + { + + if err := t.PieceBlockLocations[i].UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PieceBlockLocations[i]: %w", err) + } + + } + + } + } + return nil +} diff --git a/markets/piecestore/types.go b/markets/piecestore/types.go new file mode 100644 index 000000000..b935098a7 --- /dev/null +++ b/markets/piecestore/types.go @@ -0,0 +1,70 @@ +package piecestore + +import ( + "context" + + "github.com/ipfs/go-cid" + + "github.com/filecoin-project/go-state-types/abi" + + "github.com/filecoin-project/boost/markets/shared" +) + +//go:generate cbor-gen-for --map-encoding PieceInfo DealInfo BlockLocation PieceBlockLocation CIDInfo + +// DealInfo is information about a single deal for a given piece +type DealInfo struct { + DealID abi.DealID + SectorID abi.SectorNumber + Offset abi.PaddedPieceSize + Length abi.PaddedPieceSize +} + +// BlockLocation is information about where a given block is relative to the overall piece +type BlockLocation struct { + RelOffset uint64 + BlockSize uint64 +} + +// PieceBlockLocation is block information along with the pieceCID of the piece the block +// is inside of +type PieceBlockLocation struct { + BlockLocation + PieceCID cid.Cid +} + +// CIDInfo is information about where a given CID will live inside a piece +type CIDInfo struct { + CID cid.Cid + PieceBlockLocations []PieceBlockLocation +} + +// CIDInfoUndefined is cid info with no information +var CIDInfoUndefined = CIDInfo{} + +// PieceInfo is metadata about a piece a provider may be storing based +// on its PieceCID -- so that, given a pieceCID during retrieval, the miner +// can determine how to unseal it if needed +type PieceInfo struct { + PieceCID cid.Cid + Deals []DealInfo +} + +// PieceInfoUndefined is piece info with no information +var PieceInfoUndefined = PieceInfo{} + +func (pi PieceInfo) Defined() bool { + return pi.PieceCID.Defined() || len(pi.Deals) > 0 +} + +// PieceStore is a saved database of piece info that can be modified and queried +type PieceStore interface { + Start(ctx context.Context) error + OnReady(ready shared.ReadyFunc) + AddDealForPiece(pieceCID cid.Cid, payloadCid cid.Cid, dealInfo DealInfo) error + AddPieceBlockLocations(pieceCID cid.Cid, blockLocations map[cid.Cid]BlockLocation) error + GetPieceInfo(pieceCID cid.Cid) (PieceInfo, error) + GetCIDInfo(payloadCID cid.Cid) (CIDInfo, error) + ListCidInfoKeys() ([]cid.Cid, error) + ListPieceInfoKeys() ([]cid.Cid, error) +} diff --git a/markets/piecestore/types_cbor_gen.go b/markets/piecestore/types_cbor_gen.go new file mode 100644 index 000000000..fd74ae523 --- /dev/null +++ b/markets/piecestore/types_cbor_gen.go @@ -0,0 +1,737 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package piecestore + +import ( + "fmt" + "io" + "math" + "sort" + + abi "github.com/filecoin-project/go-state-types/abi" + cid "github.com/ipfs/go-cid" + cbg "github.com/whyrusleeping/cbor-gen" + xerrors "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +func (t *PieceInfo) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.Deals ([]piecestore.DealInfo) (slice) + if len("Deals") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Deals\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Deals"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Deals")); err != nil { + return err + } + + if len(t.Deals) > cbg.MaxLength { + return xerrors.Errorf("Slice value in field t.Deals was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajArray, uint64(len(t.Deals))); err != nil { + return err + } + for _, v := range t.Deals { + if err := v.MarshalCBOR(cw); err != nil { + return err + } + } + + // t.PieceCID (cid.Cid) (struct) + if len("PieceCID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PieceCID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PieceCID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PieceCID")); err != nil { + return err + } + + if err := cbg.WriteCid(cw, t.PieceCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PieceCID: %w", err) + } + + return nil +} + +func (t *PieceInfo) UnmarshalCBOR(r io.Reader) (err error) { + *t = PieceInfo{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("PieceInfo: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Deals ([]piecestore.DealInfo) (slice) + case "Deals": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + + if extra > cbg.MaxLength { + return fmt.Errorf("t.Deals: array too large (%d)", extra) + } + + if maj != cbg.MajArray { + return fmt.Errorf("expected cbor array") + } + + if extra > 0 { + t.Deals = make([]DealInfo, extra) + } + + for i := 0; i < int(extra); i++ { + + var v DealInfo + if err := v.UnmarshalCBOR(cr); err != nil { + return err + } + + t.Deals[i] = v + } + + // t.PieceCID (cid.Cid) (struct) + case "PieceCID": + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PieceCID: %w", err) + } + + t.PieceCID = c + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *DealInfo) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{164}); err != nil { + return err + } + + // t.DealID (abi.DealID) (uint64) + if len("DealID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DealID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DealID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DealID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.DealID)); err != nil { + return err + } + + // t.Length (abi.PaddedPieceSize) (uint64) + if len("Length") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Length\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Length"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Length")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Length)); err != nil { + return err + } + + // t.Offset (abi.PaddedPieceSize) (uint64) + if len("Offset") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Offset\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Offset"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Offset")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Offset)); err != nil { + return err + } + + // t.SectorID (abi.SectorNumber) (uint64) + if len("SectorID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"SectorID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("SectorID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("SectorID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.SectorID)); err != nil { + return err + } + + return nil +} + +func (t *DealInfo) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealInfo{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("DealInfo: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.DealID (abi.DealID) (uint64) + case "DealID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.DealID = abi.DealID(extra) + + } + // t.Length (abi.PaddedPieceSize) (uint64) + case "Length": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Length = abi.PaddedPieceSize(extra) + + } + // t.Offset (abi.PaddedPieceSize) (uint64) + case "Offset": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Offset = abi.PaddedPieceSize(extra) + + } + // t.SectorID (abi.SectorNumber) (uint64) + case "SectorID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.SectorID = abi.SectorNumber(extra) + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *BlockLocation) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.BlockSize (uint64) (uint64) + if len("BlockSize") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"BlockSize\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("BlockSize"))); err != nil { + return err + } + if _, err := cw.WriteString(string("BlockSize")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.BlockSize)); err != nil { + return err + } + + // t.RelOffset (uint64) (uint64) + if len("RelOffset") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"RelOffset\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("RelOffset"))); err != nil { + return err + } + if _, err := cw.WriteString(string("RelOffset")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.RelOffset)); err != nil { + return err + } + + return nil +} + +func (t *BlockLocation) UnmarshalCBOR(r io.Reader) (err error) { + *t = BlockLocation{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("BlockLocation: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.BlockSize (uint64) (uint64) + case "BlockSize": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.BlockSize = uint64(extra) + + } + // t.RelOffset (uint64) (uint64) + case "RelOffset": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.RelOffset = uint64(extra) + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *PieceBlockLocation) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.PieceCID (cid.Cid) (struct) + if len("PieceCID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PieceCID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PieceCID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PieceCID")); err != nil { + return err + } + + if err := cbg.WriteCid(cw, t.PieceCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PieceCID: %w", err) + } + + // t.BlockLocation (piecestore.BlockLocation) (struct) + if len("BlockLocation") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"BlockLocation\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("BlockLocation"))); err != nil { + return err + } + if _, err := cw.WriteString(string("BlockLocation")); err != nil { + return err + } + + if err := t.BlockLocation.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *PieceBlockLocation) UnmarshalCBOR(r io.Reader) (err error) { + *t = PieceBlockLocation{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("PieceBlockLocation: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.PieceCID (cid.Cid) (struct) + case "PieceCID": + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PieceCID: %w", err) + } + + t.PieceCID = c + + } + // t.BlockLocation (piecestore.BlockLocation) (struct) + case "BlockLocation": + + { + + if err := t.BlockLocation.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.BlockLocation: %w", err) + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *CIDInfo) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.CID (cid.Cid) (struct) + if len("CID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"CID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("CID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("CID")); err != nil { + return err + } + + if err := cbg.WriteCid(cw, t.CID); err != nil { + return xerrors.Errorf("failed to write cid field t.CID: %w", err) + } + + // t.PieceBlockLocations ([]piecestore.PieceBlockLocation) (slice) + if len("PieceBlockLocations") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PieceBlockLocations\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PieceBlockLocations"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PieceBlockLocations")); err != nil { + return err + } + + if len(t.PieceBlockLocations) > cbg.MaxLength { + return xerrors.Errorf("Slice value in field t.PieceBlockLocations was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajArray, uint64(len(t.PieceBlockLocations))); err != nil { + return err + } + for _, v := range t.PieceBlockLocations { + if err := v.MarshalCBOR(cw); err != nil { + return err + } + } + return nil +} + +func (t *CIDInfo) UnmarshalCBOR(r io.Reader) (err error) { + *t = CIDInfo{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("CIDInfo: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.CID (cid.Cid) (struct) + case "CID": + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.CID: %w", err) + } + + t.CID = c + + } + // t.PieceBlockLocations ([]piecestore.PieceBlockLocation) (slice) + case "PieceBlockLocations": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + + if extra > cbg.MaxLength { + return fmt.Errorf("t.PieceBlockLocations: array too large (%d)", extra) + } + + if maj != cbg.MajArray { + return fmt.Errorf("expected cbor array") + } + + if extra > 0 { + t.PieceBlockLocations = make([]PieceBlockLocation, extra) + } + + for i := 0; i < int(extra); i++ { + + var v PieceBlockLocation + if err := v.UnmarshalCBOR(cr); err != nil { + return err + } + + t.PieceBlockLocations[i] = v + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} diff --git a/markets/pricing/cli.go b/markets/pricing/cli.go deleted file mode 100644 index a3844d731..000000000 --- a/markets/pricing/cli.go +++ /dev/null @@ -1,50 +0,0 @@ -package pricing - -import ( - "bytes" - "context" - "encoding/json" - "os/exec" - - "golang.org/x/xerrors" - - "github.com/filecoin-project/boost-gfm/retrievalmarket" - - "github.com/filecoin-project/boost/node/modules/dtypes" -) - -func ExternalRetrievalPricingFunc(cmd string) dtypes.RetrievalPricingFunc { - return func(ctx context.Context, pricingInput retrievalmarket.PricingInput) (retrievalmarket.Ask, error) { - return runPricingFunc(ctx, cmd, pricingInput) - } -} - -func runPricingFunc(_ context.Context, cmd string, params interface{}) (retrievalmarket.Ask, error) { - j, err := json.Marshal(params) - if err != nil { - return retrievalmarket.Ask{}, err - } - - var out bytes.Buffer - var errb bytes.Buffer - - c := exec.Command("sh", "-c", cmd) - c.Stdin = bytes.NewReader(j) - c.Stdout = &out - c.Stderr = &errb - - switch err := c.Run().(type) { - case nil: - bz := out.Bytes() - resp := retrievalmarket.Ask{} - - if err := json.Unmarshal(bz, &resp); err != nil { - return resp, xerrors.Errorf("failed to parse pricing output %s, err=%w", string(bz), err) - } - return resp, nil - case *exec.ExitError: - return retrievalmarket.Ask{}, xerrors.Errorf("pricing func exited with error: %s", errb.String()) - default: - return retrievalmarket.Ask{}, xerrors.Errorf("pricing func cmd run error: %w", err) - } -} diff --git a/markets/retrievaladapter/client.go b/markets/retrievaladapter/client.go deleted file mode 100644 index 5cbd9f59a..000000000 --- a/markets/retrievaladapter/client.go +++ /dev/null @@ -1,127 +0,0 @@ -package retrievaladapter - -import ( - "context" - - "github.com/ipfs/go-cid" - "github.com/multiformats/go-multiaddr" - - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/boost-gfm/shared" - "github.com/filecoin-project/go-address" - "github.com/filecoin-project/go-state-types/abi" - paychtypes "github.com/filecoin-project/go-state-types/builtin/v8/paych" - - "github.com/filecoin-project/lotus/api" - "github.com/filecoin-project/lotus/chain/types" - "github.com/filecoin-project/lotus/node/impl/full" - payapi "github.com/filecoin-project/lotus/node/impl/paych" -) - -type retrievalClientNode struct { - forceOffChain bool - - chainAPI full.ChainAPI - payAPI payapi.PaychAPI - stateAPI full.StateAPI -} - -// NewRetrievalClientNode returns a new node adapter for a retrieval client that talks to the -// Lotus Node -func NewRetrievalClientNode(forceOffChain bool, payAPI payapi.PaychAPI, chainAPI full.ChainAPI, stateAPI full.StateAPI) retrievalmarket.RetrievalClientNode { - return &retrievalClientNode{ - forceOffChain: forceOffChain, - chainAPI: chainAPI, - payAPI: payAPI, - stateAPI: stateAPI, - } -} - -// GetOrCreatePaymentChannel sets up a new payment channel if one does not exist -// between a client and a miner and ensures the client has the given amount of -// funds available in the channel. -func (rcn *retrievalClientNode) GetOrCreatePaymentChannel(ctx context.Context, clientAddress address.Address, minerAddress address.Address, clientFundsAvailable abi.TokenAmount, tok shared.TipSetToken) (address.Address, cid.Cid, error) { - // TODO: respect the provided TipSetToken (a serialized TipSetKey) when - // querying the chain - ci, err := rcn.payAPI.PaychGet(ctx, clientAddress, minerAddress, clientFundsAvailable, api.PaychGetOpts{ - OffChain: rcn.forceOffChain, - }) - if err != nil { - log.Errorw("paych get failed", "error", err) - return address.Undef, cid.Undef, err - } - - return ci.Channel, ci.WaitSentinel, nil -} - -// Allocate late creates a lane within a payment channel so that calls to -// CreatePaymentVoucher will automatically make vouchers only for the difference -// in total -func (rcn *retrievalClientNode) AllocateLane(ctx context.Context, paymentChannel address.Address) (uint64, error) { - return rcn.payAPI.PaychAllocateLane(ctx, paymentChannel) -} - -// CreatePaymentVoucher creates a new payment voucher in the given lane for a -// given payment channel so that all the payment vouchers in the lane add up -// to the given amount (so the payment voucher will be for the difference) -func (rcn *retrievalClientNode) CreatePaymentVoucher(ctx context.Context, paymentChannel address.Address, amount abi.TokenAmount, lane uint64, tok shared.TipSetToken) (*paychtypes.SignedVoucher, error) { - // TODO: respect the provided TipSetToken (a serialized TipSetKey) when - // querying the chain - voucher, err := rcn.payAPI.PaychVoucherCreate(ctx, paymentChannel, amount, lane) - if err != nil { - return nil, err - } - if voucher.Voucher == nil { - return nil, retrievalmarket.NewShortfallError(voucher.Shortfall) - } - return voucher.Voucher, nil -} - -func (rcn *retrievalClientNode) GetChainHead(ctx context.Context) (shared.TipSetToken, abi.ChainEpoch, error) { - head, err := rcn.chainAPI.ChainHead(ctx) - if err != nil { - return nil, 0, err - } - - return head.Key().Bytes(), head.Height(), nil -} - -func (rcn *retrievalClientNode) WaitForPaymentChannelReady(ctx context.Context, messageCID cid.Cid) (address.Address, error) { - return rcn.payAPI.PaychGetWaitReady(ctx, messageCID) -} - -func (rcn *retrievalClientNode) CheckAvailableFunds(ctx context.Context, paymentChannel address.Address) (retrievalmarket.ChannelAvailableFunds, error) { - - channelAvailableFunds, err := rcn.payAPI.PaychAvailableFunds(ctx, paymentChannel) - if err != nil { - return retrievalmarket.ChannelAvailableFunds{}, err - } - return retrievalmarket.ChannelAvailableFunds{ - ConfirmedAmt: channelAvailableFunds.ConfirmedAmt, - PendingAmt: channelAvailableFunds.PendingAmt, - PendingWaitSentinel: channelAvailableFunds.PendingWaitSentinel, - QueuedAmt: channelAvailableFunds.QueuedAmt, - VoucherReedeemedAmt: channelAvailableFunds.VoucherReedeemedAmt, - }, nil -} - -func (rcn *retrievalClientNode) GetKnownAddresses(ctx context.Context, p retrievalmarket.RetrievalPeer, encodedTs shared.TipSetToken) ([]multiaddr.Multiaddr, error) { - tsk, err := types.TipSetKeyFromBytes(encodedTs) - if err != nil { - return nil, err - } - mi, err := rcn.stateAPI.StateMinerInfo(ctx, p.Address, tsk) - if err != nil { - return nil, err - } - multiaddrs := make([]multiaddr.Multiaddr, 0, len(mi.Multiaddrs)) - for _, a := range mi.Multiaddrs { - maddr, err := multiaddr.NewMultiaddrBytes(a) - if err != nil { - return nil, err - } - multiaddrs = append(multiaddrs, maddr) - } - - return multiaddrs, nil -} diff --git a/markets/retrievaladapter/client_blockstore.go b/markets/retrievaladapter/client_blockstore.go deleted file mode 100644 index 409f8a03a..000000000 --- a/markets/retrievaladapter/client_blockstore.go +++ /dev/null @@ -1,83 +0,0 @@ -package retrievaladapter - -import ( - "fmt" - "path/filepath" - "sync" - - bstore "github.com/ipfs/boxo/blockstore" - "github.com/ipfs/go-cid" - "github.com/ipld/go-car/v2/blockstore" - - "github.com/filecoin-project/boost-gfm/retrievalmarket" -) - -// ProxyBlockstoreAccessor is an accessor that returns a fixed blockstore. -// To be used in combination with IPFS integration. -type ProxyBlockstoreAccessor struct { - Blockstore bstore.Blockstore -} - -var _ retrievalmarket.BlockstoreAccessor = (*ProxyBlockstoreAccessor)(nil) - -func NewFixedBlockstoreAccessor(bs bstore.Blockstore) retrievalmarket.BlockstoreAccessor { - return &ProxyBlockstoreAccessor{Blockstore: bs} -} - -func (p *ProxyBlockstoreAccessor) Get(_ retrievalmarket.DealID, _ retrievalmarket.PayloadCID) (bstore.Blockstore, error) { - return p.Blockstore, nil -} - -func (p *ProxyBlockstoreAccessor) Done(_ retrievalmarket.DealID) error { - return nil -} - -type CARBlockstoreAccessor struct { - rootdir string - lk sync.Mutex - open map[retrievalmarket.DealID]*blockstore.ReadWrite -} - -var _ retrievalmarket.BlockstoreAccessor = (*CARBlockstoreAccessor)(nil) - -func NewCARBlockstoreAccessor(rootdir string) *CARBlockstoreAccessor { - return &CARBlockstoreAccessor{ - rootdir: rootdir, - open: make(map[retrievalmarket.DealID]*blockstore.ReadWrite), - } -} - -func (c *CARBlockstoreAccessor) Get(id retrievalmarket.DealID, payloadCid retrievalmarket.PayloadCID) (bstore.Blockstore, error) { - c.lk.Lock() - defer c.lk.Unlock() - - bs, ok := c.open[id] - if ok { - return bs, nil - } - - path := c.PathFor(id) - bs, err := blockstore.OpenReadWrite(path, []cid.Cid{payloadCid}, blockstore.UseWholeCIDs(true)) - if err != nil { - return nil, err - } - c.open[id] = bs - return bs, nil -} - -func (c *CARBlockstoreAccessor) Done(id retrievalmarket.DealID) error { - c.lk.Lock() - defer c.lk.Unlock() - - bs, ok := c.open[id] - if !ok { - return nil - } - - delete(c.open, id) - return bs.Finalize() -} - -func (c *CARBlockstoreAccessor) PathFor(id retrievalmarket.DealID) string { - return filepath.Join(c.rootdir, fmt.Sprintf("%d.car", id)) -} diff --git a/markets/retrievaladapter/provider.go b/markets/retrievaladapter/provider.go deleted file mode 100644 index 7a4fc44c9..000000000 --- a/markets/retrievaladapter/provider.go +++ /dev/null @@ -1,108 +0,0 @@ -package retrievaladapter - -import ( - "context" - - "github.com/hashicorp/go-multierror" - "github.com/ipfs/go-cid" - logging "github.com/ipfs/go-log/v2" - "golang.org/x/xerrors" - - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/boost-gfm/shared" - "github.com/filecoin-project/go-address" - "github.com/filecoin-project/go-state-types/abi" - paychtypes "github.com/filecoin-project/go-state-types/builtin/v8/paych" - - "github.com/filecoin-project/lotus/api/v1api" - "github.com/filecoin-project/lotus/chain/types" -) - -var log = logging.Logger("retrievaladapter") - -type retrievalProviderNode struct { - full v1api.FullNode -} - -var _ retrievalmarket.RetrievalProviderNode = (*retrievalProviderNode)(nil) - -// NewRetrievalProviderNode returns a new node adapter for a retrieval provider that talks to the -// Lotus Node -func NewRetrievalProviderNode(full v1api.FullNode) retrievalmarket.RetrievalProviderNode { - return &retrievalProviderNode{full: full} -} - -func (rpn *retrievalProviderNode) GetMinerWorkerAddress(ctx context.Context, miner address.Address, tok shared.TipSetToken) (address.Address, error) { - tsk, err := types.TipSetKeyFromBytes(tok) - if err != nil { - return address.Undef, err - } - - mi, err := rpn.full.StateMinerInfo(ctx, miner, tsk) - return mi.Worker, err -} - -func (rpn *retrievalProviderNode) SavePaymentVoucher(ctx context.Context, paymentChannel address.Address, voucher *paychtypes.SignedVoucher, proof []byte, expectedAmount abi.TokenAmount, tok shared.TipSetToken) (abi.TokenAmount, error) { - // TODO: respect the provided TipSetToken (a serialized TipSetKey) when - // querying the chain - added, err := rpn.full.PaychVoucherAdd(ctx, paymentChannel, voucher, proof, expectedAmount) - return added, err -} - -func (rpn *retrievalProviderNode) GetChainHead(ctx context.Context) (shared.TipSetToken, abi.ChainEpoch, error) { - head, err := rpn.full.ChainHead(ctx) - if err != nil { - return nil, 0, err - } - - return head.Key().Bytes(), head.Height(), nil -} - -// GetRetrievalPricingInput takes a set of candidate storage deals that can serve a retrieval request, -// and returns an minimally populated PricingInput. This PricingInput should be enhanced -// with more data, and passed to the pricing function to determine the final quoted price. -func (rpn *retrievalProviderNode) GetRetrievalPricingInput(ctx context.Context, pieceCID cid.Cid, storageDeals []abi.DealID) (retrievalmarket.PricingInput, error) { - resp := retrievalmarket.PricingInput{} - - head, err := rpn.full.ChainHead(ctx) - if err != nil { - return resp, xerrors.Errorf("failed to get chain head: %w", err) - } - tsk := head.Key() - - var mErr error - - for _, dealID := range storageDeals { - ds, err := rpn.full.StateMarketStorageDeal(ctx, dealID, tsk) - if err != nil { - log.Warnf("failed to look up deal %d on chain: err=%w", dealID, err) - mErr = multierror.Append(mErr, err) - continue - } - if ds.Proposal.VerifiedDeal { - resp.VerifiedDeal = true - } - - if ds.Proposal.PieceCID.Equals(pieceCID) { - resp.PieceSize = ds.Proposal.PieceSize.Unpadded() - } - - // If we've discovered a verified deal with the required PieceCID, we don't need - // to lookup more deals and we're done. - if resp.VerifiedDeal && resp.PieceSize != 0 { - break - } - } - - // Note: The piece size can never actually be zero. We only use it to here - // to assert that we didn't find a matching piece. - if resp.PieceSize == 0 { - if mErr == nil { - return resp, xerrors.New("failed to find matching piece") - } - - return resp, xerrors.Errorf("failed to fetch storage deal state: %w", mErr) - } - - return resp, nil -} diff --git a/markets/retrievaladapter/provider_test.go b/markets/retrievaladapter/provider_test.go deleted file mode 100644 index 71cbcee9f..000000000 --- a/markets/retrievaladapter/provider_test.go +++ /dev/null @@ -1,206 +0,0 @@ -// stm: #unit -package retrievaladapter - -import ( - "context" - "testing" - - "github.com/golang/mock/gomock" - "github.com/ipfs/go-cid" - "github.com/stretchr/testify/require" - "golang.org/x/xerrors" - - "github.com/filecoin-project/boost-gfm/retrievalmarket" - testnet "github.com/filecoin-project/boost-gfm/shared_testutil" - "github.com/filecoin-project/go-state-types/abi" - - "github.com/filecoin-project/lotus/api" - "github.com/filecoin-project/lotus/api/mocks" - "github.com/filecoin-project/lotus/chain/actors/builtin/market" - "github.com/filecoin-project/lotus/chain/types" -) - -func TestGetPricingInput(t *testing.T) { - //stm: @CHAIN_STATE_MARKET_STORAGE_DEAL_001 - ctx := context.Background() - tsk := &types.TipSet{} - key := tsk.Key() - - pcid := testnet.GenerateCids(1)[0] - deals := []abi.DealID{1, 2} - paddedSize := abi.PaddedPieceSize(128) - unpaddedSize := paddedSize.Unpadded() - - tcs := map[string]struct { - pieceCid cid.Cid - deals []abi.DealID - fFnc func(node *mocks.MockFullNode) - - expectedErrorStr string - expectedVerified bool - expectedPieceSize abi.UnpaddedPieceSize - }{ - "error when fails to fetch chain head": { - fFnc: func(n *mocks.MockFullNode) { - n.EXPECT().ChainHead(gomock.Any()).Return(tsk, xerrors.New("chain head error")).Times(1) - }, - expectedErrorStr: "chain head error", - }, - - "error when no piece matches": { - fFnc: func(n *mocks.MockFullNode) { - out1 := &api.MarketDeal{ - Proposal: market.DealProposal{ - PieceCID: testnet.GenerateCids(1)[0], - }, - } - out2 := &api.MarketDeal{ - Proposal: market.DealProposal{ - PieceCID: testnet.GenerateCids(1)[0], - }, - } - - n.EXPECT().ChainHead(gomock.Any()).Return(tsk, nil).Times(1) - gomock.InOrder( - n.EXPECT().StateMarketStorageDeal(gomock.Any(), deals[0], key).Return(out1, nil), - n.EXPECT().StateMarketStorageDeal(gomock.Any(), deals[1], key).Return(out2, nil), - ) - - }, - expectedErrorStr: "failed to find matching piece", - }, - - "error when fails to fetch deal state": { - fFnc: func(n *mocks.MockFullNode) { - out1 := &api.MarketDeal{ - Proposal: market.DealProposal{ - PieceCID: pcid, - PieceSize: paddedSize, - }, - } - out2 := &api.MarketDeal{ - Proposal: market.DealProposal{ - PieceCID: testnet.GenerateCids(1)[0], - VerifiedDeal: true, - }, - } - - n.EXPECT().ChainHead(gomock.Any()).Return(tsk, nil).Times(1) - gomock.InOrder( - n.EXPECT().StateMarketStorageDeal(gomock.Any(), deals[0], key).Return(out1, xerrors.New("error 1")), - n.EXPECT().StateMarketStorageDeal(gomock.Any(), deals[1], key).Return(out2, xerrors.New("error 2")), - ) - - }, - expectedErrorStr: "failed to fetch storage deal state", - }, - - "verified is true even if one deal is verified and we get the correct piecesize": { - fFnc: func(n *mocks.MockFullNode) { - out1 := &api.MarketDeal{ - Proposal: market.DealProposal{ - PieceCID: pcid, - PieceSize: paddedSize, - }, - } - out2 := &api.MarketDeal{ - Proposal: market.DealProposal{ - PieceCID: testnet.GenerateCids(1)[0], - VerifiedDeal: true, - }, - } - - n.EXPECT().ChainHead(gomock.Any()).Return(tsk, nil).Times(1) - gomock.InOrder( - n.EXPECT().StateMarketStorageDeal(gomock.Any(), deals[0], key).Return(out1, nil), - n.EXPECT().StateMarketStorageDeal(gomock.Any(), deals[1], key).Return(out2, nil), - ) - - }, - expectedPieceSize: unpaddedSize, - expectedVerified: true, - }, - - "success even if one deal state fetch errors out but the other deal is verified and has the required piececid": { - fFnc: func(n *mocks.MockFullNode) { - out1 := &api.MarketDeal{ - Proposal: market.DealProposal{ - PieceCID: testnet.GenerateCids(1)[0], - }, - } - out2 := &api.MarketDeal{ - Proposal: market.DealProposal{ - PieceCID: pcid, - PieceSize: paddedSize, - VerifiedDeal: true, - }, - } - - n.EXPECT().ChainHead(gomock.Any()).Return(tsk, nil).Times(1) - gomock.InOrder( - n.EXPECT().StateMarketStorageDeal(gomock.Any(), deals[0], key).Return(out1, xerrors.New("some error")), - n.EXPECT().StateMarketStorageDeal(gomock.Any(), deals[1], key).Return(out2, nil), - ) - - }, - expectedPieceSize: unpaddedSize, - expectedVerified: true, - }, - - "verified is false if both deals are unverified and we get the correct piece size": { - fFnc: func(n *mocks.MockFullNode) { - out1 := &api.MarketDeal{ - Proposal: market.DealProposal{ - PieceCID: pcid, - PieceSize: paddedSize, - VerifiedDeal: false, - }, - } - out2 := &api.MarketDeal{ - Proposal: market.DealProposal{ - PieceCID: testnet.GenerateCids(1)[0], - VerifiedDeal: false, - }, - } - - n.EXPECT().ChainHead(gomock.Any()).Return(tsk, nil).Times(1) - gomock.InOrder( - n.EXPECT().StateMarketStorageDeal(gomock.Any(), deals[0], key).Return(out1, nil), - n.EXPECT().StateMarketStorageDeal(gomock.Any(), deals[1], key).Return(out2, nil), - ) - - }, - expectedPieceSize: unpaddedSize, - expectedVerified: false, - }, - } - - for name, tc := range tcs { - tc := tc - t.Run(name, func(t *testing.T) { - mockCtrl := gomock.NewController(t) - // when test is done, assert expectations on all mock objects. - defer mockCtrl.Finish() - - mockFull := mocks.NewMockFullNode(mockCtrl) - rpn := &retrievalProviderNode{ - full: mockFull, - } - if tc.fFnc != nil { - tc.fFnc(mockFull) - } - - resp, err := rpn.GetRetrievalPricingInput(ctx, pcid, deals) - - if tc.expectedErrorStr != "" { - require.Error(t, err) - require.Contains(t, err.Error(), tc.expectedErrorStr) - require.Equal(t, retrievalmarket.PricingInput{}, resp) - } else { - require.NoError(t, err) - require.Equal(t, tc.expectedPieceSize, resp.PieceSize) - require.Equal(t, tc.expectedVerified, resp.VerifiedDeal) - } - }) - } -} diff --git a/markets/sectoraccessor/sectoraccessor.go b/markets/sectoraccessor/sectoraccessor.go index 307136c0c..b7e350f14 100644 --- a/markets/sectoraccessor/sectoraccessor.go +++ b/markets/sectoraccessor/sectoraccessor.go @@ -8,7 +8,7 @@ import ( logging "github.com/ipfs/go-log/v2" "golang.org/x/xerrors" - "github.com/filecoin-project/boost-gfm/retrievalmarket" + retrievalmarket_types "github.com/filecoin-project/boost/retrievalmarket/types" "github.com/filecoin-project/dagstore/mount" "github.com/filecoin-project/go-address" "github.com/filecoin-project/go-state-types/abi" @@ -32,7 +32,7 @@ type sectorAccessor struct { full v1api.FullNode } -var _ retrievalmarket.SectorAccessor = (*sectorAccessor)(nil) +var _ retrievalmarket_types.SectorAccessor = (*sectorAccessor)(nil) func NewSectorAccessor(maddr dtypes.MinerAddress, secb sectorblocks.SectorBuilder, pp sealer.PieceProvider, full v1api.FullNode) dagstore.SectorAccessor { return §orAccessor{address.Address(maddr), secb, pp, full} diff --git a/markets/shared/ready.go b/markets/shared/ready.go new file mode 100644 index 000000000..9b85126bb --- /dev/null +++ b/markets/shared/ready.go @@ -0,0 +1,104 @@ +package shared + +import ( + "context" + "errors" + "sync" + + "github.com/hannahhoward/go-pubsub" +) + +// ReadyFunc is function that gets called once when an event is ready +type ReadyFunc func(error) + +// ReadyDispatcher is just an pubsub dispatcher where the callback is ReadyFunc +func ReadyDispatcher(evt pubsub.Event, fn pubsub.SubscriberFn) error { + migrateErr, ok := evt.(error) + if !ok && evt != nil { + return errors.New("wrong type of event") + } + cb, ok := fn.(ReadyFunc) + if !ok { + return errors.New("wrong type of event") + } + cb(migrateErr) + return nil +} + +// ReadyManager managers listeners for a ready event +type ReadyManager struct { + ctx context.Context + Stop context.CancelFunc + + lk sync.RWMutex + isReady bool + initErr error + pubsub *pubsub.PubSub +} + +func NewReadyManager() *ReadyManager { + ctx, stop := context.WithCancel(context.Background()) + return &ReadyManager{ + ctx: ctx, + Stop: stop, + pubsub: pubsub.New(ReadyDispatcher), + } +} + +// FireReady is called when the ready event occurs +func (m *ReadyManager) FireReady(err error) error { + m.lk.Lock() + defer m.lk.Unlock() + + if m.isReady { + return nil + } + + m.isReady = true + m.initErr = err + return m.pubsub.Publish(err) +} + +// OnReady registers a listener for the ready event. +// If the event has already been fired, the callback is immediately called back +// (in a go-routine). +func (m *ReadyManager) OnReady(ready ReadyFunc) { + m.lk.Lock() + defer m.lk.Unlock() + + if m.isReady { + initErr := m.initErr + go ready(initErr) + return + } + + m.pubsub.Subscribe(ready) +} + +// AwaitReady blocks until the ready event fires. +// Returns immediately if the event already fired. +func (m *ReadyManager) AwaitReady() error { + m.lk.RLock() + isReady := m.isReady + m.lk.RUnlock() + + if isReady { + return m.initErr + } + + errch := make(chan error) + m.OnReady(func(err error) { + select { + case <-m.ctx.Done(): + errch <- m.ctx.Err() + case errch <- err: + } + }) + + select { + case <-m.ctx.Done(): + return m.ctx.Err() + case err := <-errch: + return err + } +} diff --git a/markets/shared/retrystream.go b/markets/shared/retrystream.go new file mode 100644 index 000000000..34a75d08b --- /dev/null +++ b/markets/shared/retrystream.go @@ -0,0 +1,103 @@ +package shared + +import ( + "context" + "time" + + logging "github.com/ipfs/go-log/v2" + "github.com/jpillora/backoff" + "github.com/libp2p/go-libp2p/core/network" + "github.com/libp2p/go-libp2p/core/peer" + "github.com/libp2p/go-libp2p/core/protocol" + "golang.org/x/xerrors" +) + +var log = logging.Logger("data_transfer_network") + +// The max number of attempts to open a stream +const defaultMaxStreamOpenAttempts = 5 + +// The min backoff time between retries +const defaultMinAttemptDuration = 1 * time.Second + +// The max backoff time between retries +const defaultMaxAttemptDuration = 5 * time.Minute + +// The multiplier in the backoff time for each retry +const defaultBackoffFactor = 5 + +type StreamOpener interface { + NewStream(ctx context.Context, p peer.ID, pids ...protocol.ID) (network.Stream, error) +} + +type RetryStreamOption func(*RetryStream) + +// RetryParameters changes the default parameters around connection reopening +func RetryParameters(minDuration time.Duration, maxDuration time.Duration, attempts float64, backoffFactor float64) RetryStreamOption { + return func(impl *RetryStream) { + impl.maxStreamOpenAttempts = attempts + impl.minAttemptDuration = minDuration + impl.maxAttemptDuration = maxDuration + } +} + +type RetryStream struct { + opener StreamOpener + + backoffFactor float64 + maxStreamOpenAttempts float64 + minAttemptDuration time.Duration + maxAttemptDuration time.Duration +} + +func NewRetryStream(opener StreamOpener, options ...RetryStreamOption) *RetryStream { + impl := &RetryStream{ + opener: opener, + backoffFactor: defaultBackoffFactor, + maxStreamOpenAttempts: defaultMaxStreamOpenAttempts, + minAttemptDuration: defaultMinAttemptDuration, + maxAttemptDuration: defaultMaxAttemptDuration, + } + impl.SetOptions(options...) + return impl +} + +func (impl *RetryStream) SetOptions(options ...RetryStreamOption) { + for _, option := range options { + option(impl) + } +} + +func (impl *RetryStream) OpenStream(ctx context.Context, id peer.ID, protocols []protocol.ID) (network.Stream, error) { + b := &backoff.Backoff{ + Min: impl.minAttemptDuration, + Max: impl.maxAttemptDuration, + Factor: impl.maxStreamOpenAttempts, + Jitter: true, + } + + for { + s, err := impl.opener.NewStream(ctx, id, protocols...) + if err == nil { + return s, err + } + + // b.Attempt() starts from zero + nAttempts := b.Attempt() + 1 + if nAttempts >= impl.maxStreamOpenAttempts { + return nil, xerrors.Errorf("exhausted %d attempts but failed to open stream, err: %w", int(impl.maxStreamOpenAttempts), err) + } + + duration := b.Duration() + log.Warnf("failed to open stream to %s on attempt %.0f of %.0f, waiting %s to try again, err: %s", + id, nAttempts, impl.maxStreamOpenAttempts, duration, err) + + ebt := time.NewTimer(duration) + select { + case <-ctx.Done(): + ebt.Stop() + return nil, xerrors.Errorf("open stream to %s canceled by context", id) + case <-ebt.C: + } + } +} diff --git a/markets/shared/shared.go b/markets/shared/shared.go new file mode 100644 index 000000000..7f752a236 --- /dev/null +++ b/markets/shared/shared.go @@ -0,0 +1,7 @@ +package shared + +// TipSetToken is the implementation-nonspecific identity for a tipset. +type TipSetToken []byte + +// Unsubscribe is a function that gets called to unsubscribe from (storage|retrieval)market events +type Unsubscribe func() diff --git a/markets/shared/timecounter.go b/markets/shared/timecounter.go new file mode 100644 index 000000000..68ca73bc0 --- /dev/null +++ b/markets/shared/timecounter.go @@ -0,0 +1,21 @@ +package shared + +import ( + "sync/atomic" + "time" +) + +// TimeCounter is used to generate a monotonically increasing sequence. +// It starts at the current time, then increments on each call to next. +type TimeCounter struct { + counter uint64 +} + +func NewTimeCounter() *TimeCounter { + return &TimeCounter{counter: uint64(time.Now().UnixNano())} +} + +func (tc *TimeCounter) Next() uint64 { + counter := atomic.AddUint64(&tc.counter, 1) + return counter +} diff --git a/markets/storageadapter/api.go b/markets/storageadapter/api.go deleted file mode 100644 index 922585825..000000000 --- a/markets/storageadapter/api.go +++ /dev/null @@ -1,49 +0,0 @@ -package storageadapter - -import ( - "context" - - "github.com/filecoin-project/go-address" - "github.com/filecoin-project/lotus/blockstore" - "github.com/filecoin-project/lotus/chain/actors/adt" - "github.com/filecoin-project/lotus/chain/actors/builtin/miner" - "github.com/filecoin-project/lotus/chain/types" - cbor "github.com/ipfs/go-ipld-cbor" - "golang.org/x/xerrors" -) - -type apiWrapper struct { - api interface { - StateGetActor(ctx context.Context, actor address.Address, tsk types.TipSetKey) (*types.Actor, error) - blockstore.ChainIO - } -} - -func (ca *apiWrapper) diffPreCommits(ctx context.Context, actor address.Address, pre, cur types.TipSetKey) (*miner.PreCommitChanges, error) { - store := adt.WrapStore(ctx, cbor.NewCborStore(blockstore.NewAPIBlockstore(ca.api))) - - preAct, err := ca.api.StateGetActor(ctx, actor, pre) - if err != nil { - return nil, xerrors.Errorf("getting pre actor: %w", err) - } - curAct, err := ca.api.StateGetActor(ctx, actor, cur) - if err != nil { - return nil, xerrors.Errorf("getting cur actor: %w", err) - } - - preSt, err := miner.Load(store, preAct) - if err != nil { - return nil, xerrors.Errorf("loading miner actor: %w", err) - } - curSt, err := miner.Load(store, curAct) - if err != nil { - return nil, xerrors.Errorf("loading miner actor: %w", err) - } - - diff, err := miner.DiffPreCommits(preSt, curSt) - if err != nil { - return nil, xerrors.Errorf("diff precommits: %w", err) - } - - return diff, err -} diff --git a/markets/storageadapter/client.go b/markets/storageadapter/client.go deleted file mode 100644 index 2b3424c42..000000000 --- a/markets/storageadapter/client.go +++ /dev/null @@ -1,445 +0,0 @@ -package storageadapter - -// this file implements storagemarket.StorageClientNode - -import ( - "bytes" - "context" - - "github.com/ipfs/go-cid" - "go.uber.org/fx" - "golang.org/x/xerrors" - - "github.com/filecoin-project/boost-gfm/shared" - "github.com/filecoin-project/boost-gfm/storagemarket" - "github.com/filecoin-project/boost/markets/utils" - "github.com/filecoin-project/go-address" - cborutil "github.com/filecoin-project/go-cbor-util" - "github.com/filecoin-project/go-state-types/abi" - "github.com/filecoin-project/go-state-types/big" - "github.com/filecoin-project/go-state-types/builtin" - markettypes "github.com/filecoin-project/go-state-types/builtin/v9/market" - "github.com/filecoin-project/go-state-types/crypto" - "github.com/filecoin-project/go-state-types/exitcode" - "github.com/filecoin-project/lotus/api" - "github.com/filecoin-project/lotus/build" - marketactor "github.com/filecoin-project/lotus/chain/actors/builtin/market" - "github.com/filecoin-project/lotus/chain/events" - "github.com/filecoin-project/lotus/chain/events/state" - "github.com/filecoin-project/lotus/chain/market" - "github.com/filecoin-project/lotus/chain/types" - "github.com/filecoin-project/lotus/lib/sigs" - "github.com/filecoin-project/lotus/node/impl/full" - "github.com/filecoin-project/lotus/node/modules/helpers" -) - -type ClientNodeAdapter struct { - *clientApi - - fundmgr *market.FundManager - ev *events.Events - dsMatcher *dealStateMatcher - scMgr *SectorCommittedManager -} - -type clientApi struct { - full.ChainAPI - full.StateAPI - full.MpoolAPI -} - -func NewClientNodeAdapter(mctx helpers.MetricsCtx, lc fx.Lifecycle, stateapi full.StateAPI, chain full.ChainAPI, mpool full.MpoolAPI, fundmgr *market.FundManager) (storagemarket.StorageClientNode, error) { - capi := &clientApi{chain, stateapi, mpool} - ctx := helpers.LifecycleCtx(mctx, lc) - - ev, err := events.NewEvents(ctx, capi) - if err != nil { - return nil, err - } - a := &ClientNodeAdapter{ - clientApi: capi, - - fundmgr: fundmgr, - ev: ev, - dsMatcher: newDealStateMatcher(state.NewStatePredicates(state.WrapFastAPI(capi))), - } - a.scMgr = NewSectorCommittedManager(ev, a, &apiWrapper{api: capi}) - return a, nil -} - -func (c *ClientNodeAdapter) ListStorageProviders(ctx context.Context, encodedTs shared.TipSetToken) ([]*storagemarket.StorageProviderInfo, error) { - tsk, err := types.TipSetKeyFromBytes(encodedTs) - if err != nil { - return nil, err - } - - addresses, err := c.StateListMiners(ctx, tsk) - if err != nil { - return nil, err - } - - var out []*storagemarket.StorageProviderInfo - - for _, addr := range addresses { - mi, err := c.GetMinerInfo(ctx, addr, encodedTs) - if err != nil { - return nil, err - } - - out = append(out, mi) - } - - return out, nil -} - -func (c *ClientNodeAdapter) VerifySignature(ctx context.Context, sig crypto.Signature, addr address.Address, input []byte, encodedTs shared.TipSetToken) (bool, error) { - addr, err := c.StateAccountKey(ctx, addr, types.EmptyTSK) - if err != nil { - return false, err - } - - err = sigs.Verify(&sig, addr, input) - return err == nil, err -} - -// Adds funds with the StorageMinerActor for a storage participant. Used by both providers and clients. -func (c *ClientNodeAdapter) AddFunds(ctx context.Context, addr address.Address, amount abi.TokenAmount) (cid.Cid, error) { - // (Provider Node API) - smsg, err := c.MpoolPushMessage(ctx, &types.Message{ - To: marketactor.Address, - From: addr, - Value: amount, - Method: builtin.MethodsMarket.AddBalance, - }, nil) - if err != nil { - return cid.Undef, err - } - - return smsg.Cid(), nil -} - -func (c *ClientNodeAdapter) ReserveFunds(ctx context.Context, wallet, addr address.Address, amt abi.TokenAmount) (cid.Cid, error) { - return c.fundmgr.Reserve(ctx, wallet, addr, amt) -} - -func (c *ClientNodeAdapter) ReleaseFunds(ctx context.Context, addr address.Address, amt abi.TokenAmount) error { - return c.fundmgr.Release(addr, amt) -} - -func (c *ClientNodeAdapter) GetBalance(ctx context.Context, addr address.Address, encodedTs shared.TipSetToken) (storagemarket.Balance, error) { - tsk, err := types.TipSetKeyFromBytes(encodedTs) - if err != nil { - return storagemarket.Balance{}, err - } - - bal, err := c.StateMarketBalance(ctx, addr, tsk) - if err != nil { - return storagemarket.Balance{}, err - } - - return utils.ToSharedBalance(bal), nil -} - -// ValidatePublishedDeal validates that the provided deal has appeared on chain and references the same ClientDeal -// returns the Deal id if there is no error -// TODO: Don't return deal ID -func (c *ClientNodeAdapter) ValidatePublishedDeal(ctx context.Context, deal storagemarket.ClientDeal) (abi.DealID, error) { - log.Infow("DEAL ACCEPTED!") - - pubmsg, err := c.ChainGetMessage(ctx, *deal.PublishMessage) - if err != nil { - return 0, xerrors.Errorf("getting deal publish message: %w", err) - } - - mi, err := c.StateMinerInfo(ctx, deal.Proposal.Provider, types.EmptyTSK) - if err != nil { - return 0, xerrors.Errorf("getting miner worker failed: %w", err) - } - - fromid, err := c.StateLookupID(ctx, pubmsg.From, types.EmptyTSK) - if err != nil { - return 0, xerrors.Errorf("failed to resolve from msg ID addr: %w", err) - } - - var pubOk bool - pubAddrs := append([]address.Address{mi.Worker, mi.Owner}, mi.ControlAddresses...) - for _, a := range pubAddrs { - if fromid == a { - pubOk = true - break - } - } - - if !pubOk { - return 0, xerrors.Errorf("deal wasn't published by storage provider: from=%s, provider=%s,%+v", pubmsg.From, deal.Proposal.Provider, pubAddrs) - } - - if pubmsg.To != marketactor.Address { - return 0, xerrors.Errorf("deal publish message wasn't set to StorageMarket actor (to=%s)", pubmsg.To) - } - - if pubmsg.Method != builtin.MethodsMarket.PublishStorageDeals { - return 0, xerrors.Errorf("deal publish message called incorrect method (method=%s)", pubmsg.Method) - } - - var params markettypes.PublishStorageDealsParams - if err := params.UnmarshalCBOR(bytes.NewReader(pubmsg.Params)); err != nil { - return 0, err - } - - dealIdx := -1 - for i, storageDeal := range params.Deals { - // TODO: make it less hacky - sd := storageDeal - eq, err := cborutil.Equals(&deal.ClientDealProposal, &sd) - if err != nil { - return 0, err - } - if eq { - dealIdx = i - break - } - } - - if dealIdx == -1 { - return 0, xerrors.Errorf("deal publish didn't contain our deal (message cid: %s)", deal.PublishMessage) - } - - // TODO: timeout - ret, err := c.StateWaitMsg(ctx, *deal.PublishMessage, build.MessageConfidence, api.LookbackNoLimit, true) - if err != nil { - return 0, xerrors.Errorf("waiting for deal publish message: %w", err) - } - if ret.Receipt.ExitCode != 0 { - return 0, xerrors.Errorf("deal publish failed: exit=%d", ret.Receipt.ExitCode) - } - - nv, err := c.StateNetworkVersion(ctx, ret.TipSet) - if err != nil { - return 0, xerrors.Errorf("getting network version: %w", err) - } - - res, err := marketactor.DecodePublishStorageDealsReturn(ret.Receipt.Return, nv) - if err != nil { - return 0, xerrors.Errorf("decoding deal publish return: %w", err) - } - - dealIDs, err := res.DealIDs() - if err != nil { - return 0, xerrors.Errorf("getting dealIDs: %w", err) - } - - if dealIdx >= len(params.Deals) { - return 0, xerrors.Errorf( - "deal index %d out of bounds of deals (len %d) in publish deals message %s", - dealIdx, len(params.Deals), pubmsg.Cid()) - } - - valid, outIdx, err := res.IsDealValid(uint64(dealIdx)) - if err != nil { - return 0, xerrors.Errorf("determining deal validity: %w", err) - } - - if !valid { - return 0, xerrors.New("deal was invalid at publication") - } - - return dealIDs[outIdx], nil -} - -var clientOverestimation = struct { - numerator int64 - denominator int64 -}{ - numerator: 12, - denominator: 10, -} - -func (c *ClientNodeAdapter) DealProviderCollateralBounds(ctx context.Context, size abi.PaddedPieceSize, isVerified bool) (abi.TokenAmount, abi.TokenAmount, error) { - bounds, err := c.StateDealProviderCollateralBounds(ctx, size, isVerified, types.EmptyTSK) - if err != nil { - return abi.TokenAmount{}, abi.TokenAmount{}, err - } - - min := big.Mul(bounds.Min, big.NewInt(clientOverestimation.numerator)) - min = big.Div(min, big.NewInt(clientOverestimation.denominator)) - return min, bounds.Max, nil -} - -// TODO: Remove dealID parameter, change publishCid to be cid.Cid (instead of pointer) -func (c *ClientNodeAdapter) OnDealSectorPreCommitted(ctx context.Context, provider address.Address, dealID abi.DealID, proposal markettypes.DealProposal, publishCid *cid.Cid, cb storagemarket.DealSectorPreCommittedCallback) error { - return c.scMgr.OnDealSectorPreCommitted(ctx, provider, proposal, *publishCid, cb) -} - -// TODO: Remove dealID parameter, change publishCid to be cid.Cid (instead of pointer) -func (c *ClientNodeAdapter) OnDealSectorCommitted(ctx context.Context, provider address.Address, dealID abi.DealID, sectorNumber abi.SectorNumber, proposal markettypes.DealProposal, publishCid *cid.Cid, cb storagemarket.DealSectorCommittedCallback) error { - return c.scMgr.OnDealSectorCommitted(ctx, provider, sectorNumber, proposal, *publishCid, cb) -} - -// TODO: Replace dealID parameter with DealProposal -func (c *ClientNodeAdapter) OnDealExpiredOrSlashed(ctx context.Context, dealID abi.DealID, onDealExpired storagemarket.DealExpiredCallback, onDealSlashed storagemarket.DealSlashedCallback) error { - head, err := c.ChainHead(ctx) - if err != nil { - return xerrors.Errorf("client: failed to get chain head: %w", err) - } - - sd, err := c.StateMarketStorageDeal(ctx, dealID, head.Key()) - if err != nil { - return xerrors.Errorf("client: failed to look up deal %d on chain: %w", dealID, err) - } - - // Called immediately to check if the deal has already expired or been slashed - checkFunc := func(ctx context.Context, ts *types.TipSet) (done bool, more bool, err error) { - if ts == nil { - // keep listening for events - return false, true, nil - } - - // Check if the deal has already expired - if sd.Proposal.EndEpoch <= ts.Height() { - onDealExpired(nil) - return true, false, nil - } - - // If there is no deal assume it's already been slashed - if sd.State.SectorStartEpoch < 0 { - onDealSlashed(ts.Height(), nil) - return true, false, nil - } - - // No events have occurred yet, so return - // done: false, more: true (keep listening for events) - return false, true, nil - } - - // Called when there was a match against the state change we're looking for - // and the chain has advanced to the confidence height - stateChanged := func(ts *types.TipSet, ts2 *types.TipSet, states events.StateChange, h abi.ChainEpoch) (more bool, err error) { - // Check if the deal has already expired - if ts2 == nil || sd.Proposal.EndEpoch <= ts2.Height() { - onDealExpired(nil) - return false, nil - } - - // Timeout waiting for state change - if states == nil { - log.Error("timed out waiting for deal expiry") - return false, nil - } - - changedDeals, ok := states.(state.ChangedDeals) - if !ok { - panic("Expected state.ChangedDeals") - } - - deal, ok := changedDeals[dealID] - if !ok { - // No change to deal - return true, nil - } - - // Deal was slashed - if deal.To == nil { - onDealSlashed(ts2.Height(), nil) - return false, nil - } - - return true, nil - } - - // Called when there was a chain reorg and the state change was reverted - revert := func(ctx context.Context, ts *types.TipSet) error { - // TODO: Is it ok to just ignore this? - log.Warn("deal state reverted; TODO: actually handle this!") - return nil - } - - // Watch for state changes to the deal - match := c.dsMatcher.matcher(ctx, dealID) - - // Wait until after the end epoch for the deal and then timeout - timeout := (sd.Proposal.EndEpoch - head.Height()) + 1 - if err := c.ev.StateChanged(checkFunc, stateChanged, revert, int(build.MessageConfidence)+1, timeout, match); err != nil { - return xerrors.Errorf("failed to set up state changed handler: %w", err) - } - - return nil -} - -func (c *ClientNodeAdapter) SignProposal(ctx context.Context, signer address.Address, proposal markettypes.DealProposal) (*markettypes.ClientDealProposal, error) { - // TODO: output spec signed proposal - buf, err := cborutil.Dump(&proposal) - if err != nil { - return nil, err - } - - signer, err = c.StateAccountKey(ctx, signer, types.EmptyTSK) - if err != nil { - return nil, err - } - - sig, err := c.Wallet.WalletSign(ctx, signer, buf, api.MsgMeta{ - Type: api.MTDealProposal, - }) - if err != nil { - return nil, err - } - - return &markettypes.ClientDealProposal{ - Proposal: proposal, - ClientSignature: *sig, - }, nil -} - -func (c *ClientNodeAdapter) GetDefaultWalletAddress(ctx context.Context) (address.Address, error) { - addr, err := c.DefWallet.GetDefault() - return addr, err -} - -func (c *ClientNodeAdapter) GetChainHead(ctx context.Context) (shared.TipSetToken, abi.ChainEpoch, error) { - head, err := c.ChainHead(ctx) - if err != nil { - return nil, 0, err - } - - return head.Key().Bytes(), head.Height(), nil -} - -func (c *ClientNodeAdapter) WaitForMessage(ctx context.Context, mcid cid.Cid, cb func(code exitcode.ExitCode, bytes []byte, finalCid cid.Cid, err error) error) error { - receipt, err := c.StateWaitMsg(ctx, mcid, build.MessageConfidence, api.LookbackNoLimit, true) - if err != nil { - return cb(0, nil, cid.Undef, err) - } - return cb(receipt.Receipt.ExitCode, receipt.Receipt.Return, receipt.Message, nil) -} - -func (c *ClientNodeAdapter) GetMinerInfo(ctx context.Context, addr address.Address, encodedTs shared.TipSetToken) (*storagemarket.StorageProviderInfo, error) { - tsk, err := types.TipSetKeyFromBytes(encodedTs) - if err != nil { - return nil, err - } - mi, err := c.StateMinerInfo(ctx, addr, tsk) - if err != nil { - return nil, err - } - - out := utils.NewStorageProviderInfo(addr, mi.Worker, mi.SectorSize, *mi.PeerId, mi.Multiaddrs) - return &out, nil -} - -func (c *ClientNodeAdapter) SignBytes(ctx context.Context, signer address.Address, b []byte) (*crypto.Signature, error) { - signer, err := c.StateAccountKey(ctx, signer, types.EmptyTSK) - if err != nil { - return nil, err - } - - localSignature, err := c.Wallet.WalletSign(ctx, signer, b, api.MsgMeta{ - Type: api.MTUnknown, // TODO: pass type here - }) - if err != nil { - return nil, err - } - return localSignature, nil -} - -var _ storagemarket.StorageClientNode = &ClientNodeAdapter{} diff --git a/markets/storageadapter/client_blockstore.go b/markets/storageadapter/client_blockstore.go deleted file mode 100644 index 867e64493..000000000 --- a/markets/storageadapter/client_blockstore.go +++ /dev/null @@ -1,102 +0,0 @@ -package storageadapter - -import ( - "sync" - - blockstore "github.com/ipfs/boxo/blockstore" - "github.com/ipfs/go-cid" - "golang.org/x/xerrors" - - "github.com/filecoin-project/boost-gfm/storagemarket" - "github.com/filecoin-project/boost-gfm/stores" - - "github.com/filecoin-project/lotus/node/repo/imports" -) - -// ProxyBlockstoreAccessor is an accessor that returns a fixed blockstore. -// To be used in combination with IPFS integration. -type ProxyBlockstoreAccessor struct { - Blockstore blockstore.Blockstore -} - -var _ storagemarket.BlockstoreAccessor = (*ProxyBlockstoreAccessor)(nil) - -func NewFixedBlockstoreAccessor(bs blockstore.Blockstore) storagemarket.BlockstoreAccessor { - return &ProxyBlockstoreAccessor{Blockstore: bs} -} - -func (p *ProxyBlockstoreAccessor) Get(cid storagemarket.PayloadCID) (blockstore.Blockstore, error) { - return p.Blockstore, nil -} - -func (p *ProxyBlockstoreAccessor) Done(cid storagemarket.PayloadCID) error { - return nil -} - -// ImportsBlockstoreAccessor is a blockstore accessor backed by the -// imports.Manager. -type ImportsBlockstoreAccessor struct { - m *imports.Manager - lk sync.Mutex - open map[cid.Cid]struct { - st stores.ClosableBlockstore - refs int - } -} - -var _ storagemarket.BlockstoreAccessor = (*ImportsBlockstoreAccessor)(nil) - -func NewImportsBlockstoreAccessor(importmgr *imports.Manager) *ImportsBlockstoreAccessor { - return &ImportsBlockstoreAccessor{ - m: importmgr, - open: make(map[cid.Cid]struct { - st stores.ClosableBlockstore - refs int - }), - } -} - -func (s *ImportsBlockstoreAccessor) Get(payloadCID storagemarket.PayloadCID) (blockstore.Blockstore, error) { - s.lk.Lock() - defer s.lk.Unlock() - - e, ok := s.open[payloadCID] - if ok { - e.refs++ - return e.st, nil - } - - path, err := s.m.CARPathFor(payloadCID) - if err != nil { - return nil, xerrors.Errorf("failed to get client blockstore for root %s: %w", payloadCID, err) - } - if path == "" { - return nil, xerrors.Errorf("no client blockstore for root %s", payloadCID) - } - ret, err := stores.ReadOnlyFilestore(path) - if err != nil { - return nil, err - } - e.st = ret - s.open[payloadCID] = e - return ret, nil -} - -func (s *ImportsBlockstoreAccessor) Done(payloadCID storagemarket.PayloadCID) error { - s.lk.Lock() - defer s.lk.Unlock() - - e, ok := s.open[payloadCID] - if !ok { - return nil - } - - e.refs-- - if e.refs == 0 { - if err := e.st.Close(); err != nil { - log.Warnf("failed to close blockstore: %s", err) - } - delete(s.open, payloadCID) - } - return nil -} diff --git a/markets/storageadapter/dealpublisher.go b/markets/storageadapter/dealpublisher.go index 74d8a7d19..54aeb952f 100644 --- a/markets/storageadapter/dealpublisher.go +++ b/markets/storageadapter/dealpublisher.go @@ -8,7 +8,9 @@ import ( "time" cborutil "github.com/filecoin-project/go-cbor-util" + "github.com/filecoin-project/go-state-types/exitcode" "github.com/ipfs/go-cid" + logging "github.com/ipfs/go-log/v2" "go.uber.org/fx" "golang.org/x/xerrors" @@ -17,16 +19,15 @@ import ( "github.com/filecoin-project/go-state-types/big" "github.com/filecoin-project/go-state-types/builtin" "github.com/filecoin-project/go-state-types/builtin/v9/market" - "github.com/filecoin-project/go-state-types/exitcode" - "github.com/filecoin-project/lotus/api" "github.com/filecoin-project/lotus/build" "github.com/filecoin-project/lotus/chain/actors" "github.com/filecoin-project/lotus/chain/types" - "github.com/filecoin-project/lotus/node/config" "github.com/filecoin-project/lotus/storage/ctladdr" ) +var log = logging.Logger("storageadapter") + type dealPublisherAPI interface { ChainHead(context.Context) (*types.TipSet, error) MpoolPushMessage(ctx context.Context, msg *types.Message, spec *api.MessageSendSpec) (*types.SignedMessage, error) @@ -110,13 +111,13 @@ type PublishMsgConfig struct { } func NewDealPublisher( - feeConfig *config.MinerFeeConfig, + maxPublishDealsFee *types.FIL, publishMsgCfg PublishMsgConfig, ) func(lc fx.Lifecycle, full api.FullNode, as *ctladdr.AddressSelector) *DealPublisher { return func(lc fx.Lifecycle, full api.FullNode, as *ctladdr.AddressSelector) *DealPublisher { maxFee := abi.NewTokenAmount(0) - if feeConfig != nil { - maxFee = abi.TokenAmount(feeConfig.MaxPublishDealsFee) + if maxPublishDealsFee != nil { + maxFee = abi.TokenAmount(*maxPublishDealsFee) } publishSpec := &api.MessageSendSpec{MaxFee: maxFee} dp := newDealPublisher(full, as, publishMsgCfg, publishSpec) diff --git a/markets/storageadapter/dealpublisher_test.go b/markets/storageadapter/dealpublisher_test.go index 7b2b182fc..0d01c1fd2 100644 --- a/markets/storageadapter/dealpublisher_test.go +++ b/markets/storageadapter/dealpublisher_test.go @@ -4,10 +4,12 @@ package storageadapter import ( "bytes" "context" + "fmt" "testing" "time" cborutil "github.com/filecoin-project/go-cbor-util" + blocks "github.com/ipfs/go-block-format" "github.com/ipfs/go-cid" "github.com/raulk/clock" "github.com/stretchr/testify/require" @@ -483,3 +485,15 @@ func getWorkerActor(t *testing.T) address.Address { func getProviderActor(t *testing.T) address.Address { return tutils.NewActorAddr(t, "provider") } + +var seq int + +func generateCids(n int) []cid.Cid { + cids := make([]cid.Cid, 0, n) + for i := 0; i < n; i++ { + c := blocks.NewBlock([]byte(fmt.Sprint(seq))).Cid() + seq++ + cids = append(cids, c) + } + return cids +} diff --git a/markets/storageadapter/dealstatematcher.go b/markets/storageadapter/dealstatematcher.go deleted file mode 100644 index 8d5598eae..000000000 --- a/markets/storageadapter/dealstatematcher.go +++ /dev/null @@ -1,85 +0,0 @@ -package storageadapter - -import ( - "context" - "sync" - - "github.com/filecoin-project/go-state-types/abi" - - actorsmarket "github.com/filecoin-project/lotus/chain/actors/builtin/market" - "github.com/filecoin-project/lotus/chain/events" - "github.com/filecoin-project/lotus/chain/events/state" - "github.com/filecoin-project/lotus/chain/types" -) - -// dealStateMatcher caches the DealStates for the most recent -// old/new tipset combination -type dealStateMatcher struct { - preds *state.StatePredicates - - lk sync.Mutex - oldTsk types.TipSetKey - newTsk types.TipSetKey - oldDealStateRoot actorsmarket.DealStates - newDealStateRoot actorsmarket.DealStates -} - -func newDealStateMatcher(preds *state.StatePredicates) *dealStateMatcher { - return &dealStateMatcher{preds: preds} -} - -// matcher returns a function that checks if the state of the given dealID -// has changed. -// It caches the DealStates for the most recent old/new tipset combination. -func (mc *dealStateMatcher) matcher(ctx context.Context, dealID abi.DealID) events.StateMatchFunc { - // The function that is called to check if the deal state has changed for - // the target deal ID - dealStateChangedForID := mc.preds.DealStateChangedForIDs([]abi.DealID{dealID}) - - // The match function is called by the events API to check if there's - // been a state change for the deal with the target deal ID - match := func(oldTs, newTs *types.TipSet) (bool, events.StateChange, error) { - mc.lk.Lock() - defer mc.lk.Unlock() - - // Check if we've already fetched the DealStates for the given tipsets - if mc.oldTsk == oldTs.Key() && mc.newTsk == newTs.Key() { - // If we fetch the DealStates and there is no difference between - // them, they are stored as nil. So we can just bail out. - if mc.oldDealStateRoot == nil || mc.newDealStateRoot == nil { - return false, nil, nil - } - - // Check if the deal state has changed for the target ID - return dealStateChangedForID(ctx, mc.oldDealStateRoot, mc.newDealStateRoot) - } - - // We haven't already fetched the DealStates for the given tipsets, so - // do so now - - // Replace dealStateChangedForID with a function that records the - // DealStates so that we can cache them - var oldDealStateRootSaved, newDealStateRootSaved actorsmarket.DealStates - recorder := func(ctx context.Context, oldDealStateRoot, newDealStateRoot actorsmarket.DealStates) (changed bool, user state.UserData, err error) { - // Record DealStates - oldDealStateRootSaved = oldDealStateRoot - newDealStateRootSaved = newDealStateRoot - - return dealStateChangedForID(ctx, oldDealStateRoot, newDealStateRoot) - } - - // Call the match function - dealDiff := mc.preds.OnStorageMarketActorChanged( - mc.preds.OnDealStateChanged(recorder)) - matched, data, err := dealDiff(ctx, oldTs.Key(), newTs.Key()) - - // Save the recorded DealStates for the tipsets - mc.oldTsk = oldTs.Key() - mc.newTsk = newTs.Key() - mc.oldDealStateRoot = oldDealStateRootSaved - mc.newDealStateRoot = newDealStateRootSaved - - return matched, data, err - } - return match -} diff --git a/markets/storageadapter/dealstatematcher_test.go b/markets/storageadapter/dealstatematcher_test.go deleted file mode 100644 index 9a46e4af9..000000000 --- a/markets/storageadapter/dealstatematcher_test.go +++ /dev/null @@ -1,155 +0,0 @@ -// stm: #unit -package storageadapter - -import ( - "context" - "testing" - - "github.com/ipfs/go-cid" - cbornode "github.com/ipfs/go-ipld-cbor" - "github.com/stretchr/testify/require" - "golang.org/x/sync/errgroup" - - "github.com/filecoin-project/go-address" - "github.com/filecoin-project/go-state-types/abi" - builtin2 "github.com/filecoin-project/specs-actors/v2/actors/builtin" - market2 "github.com/filecoin-project/specs-actors/v2/actors/builtin/market" - adt2 "github.com/filecoin-project/specs-actors/v2/actors/util/adt" - - bstore "github.com/filecoin-project/lotus/blockstore" - "github.com/filecoin-project/lotus/chain/events" - "github.com/filecoin-project/lotus/chain/events/state" - test "github.com/filecoin-project/lotus/chain/events/state/mock" - "github.com/filecoin-project/lotus/chain/types" -) - -func TestDealStateMatcher(t *testing.T) { - //stm: @CHAIN_STATE_GET_ACTOR_001 - ctx := context.Background() - bs := bstore.NewMemorySync() - store := adt2.WrapStore(ctx, cbornode.NewCborStore(bs)) - - deal1 := &market2.DealState{ - SectorStartEpoch: 1, - LastUpdatedEpoch: 2, - } - deal2 := &market2.DealState{ - SectorStartEpoch: 4, - LastUpdatedEpoch: 5, - } - deal3 := &market2.DealState{ - SectorStartEpoch: 7, - LastUpdatedEpoch: 8, - } - deals1 := map[abi.DealID]*market2.DealState{ - abi.DealID(1): deal1, - } - deals2 := map[abi.DealID]*market2.DealState{ - abi.DealID(1): deal2, - } - deals3 := map[abi.DealID]*market2.DealState{ - abi.DealID(1): deal3, - } - - deal1StateC := createMarketState(ctx, t, store, deals1) - deal2StateC := createMarketState(ctx, t, store, deals2) - deal3StateC := createMarketState(ctx, t, store, deals3) - - minerAddr, err := address.NewFromString("t00") - require.NoError(t, err) - ts1, err := test.MockTipset(minerAddr, 1) - require.NoError(t, err) - ts2, err := test.MockTipset(minerAddr, 2) - require.NoError(t, err) - ts3, err := test.MockTipset(minerAddr, 3) - require.NoError(t, err) - - api := test.NewMockAPI(bs) - api.SetActor(ts1.Key(), &types.Actor{Code: builtin2.StorageMarketActorCodeID, Head: deal1StateC}) - api.SetActor(ts2.Key(), &types.Actor{Code: builtin2.StorageMarketActorCodeID, Head: deal2StateC}) - api.SetActor(ts3.Key(), &types.Actor{Code: builtin2.StorageMarketActorCodeID, Head: deal3StateC}) - - t.Run("caching", func(t *testing.T) { - dsm := newDealStateMatcher(state.NewStatePredicates(api)) - matcher := dsm.matcher(ctx, abi.DealID(1)) - - // Call matcher with tipsets that have the same state - ok, stateChange, err := matcher(ts1, ts1) - require.NoError(t, err) - require.False(t, ok) - require.Nil(t, stateChange) - // Should call StateGetActor once for each tipset - require.Equal(t, 2, api.StateGetActorCallCount()) - - // Call matcher with tipsets that have different state - api.ResetCallCounts() - ok, stateChange, err = matcher(ts1, ts2) - require.NoError(t, err) - require.True(t, ok) - require.NotNil(t, stateChange) - // Should call StateGetActor once for each tipset - require.Equal(t, 2, api.StateGetActorCallCount()) - - // Call matcher again with the same tipsets as above, should be cached - api.ResetCallCounts() - ok, stateChange, err = matcher(ts1, ts2) - require.NoError(t, err) - require.True(t, ok) - require.NotNil(t, stateChange) - // Should not call StateGetActor (because it should hit the cache) - require.Equal(t, 0, api.StateGetActorCallCount()) - - // Call matcher with different tipsets, should not be cached - api.ResetCallCounts() - ok, stateChange, err = matcher(ts2, ts3) - require.NoError(t, err) - require.True(t, ok) - require.NotNil(t, stateChange) - // Should call StateGetActor once for each tipset - require.Equal(t, 2, api.StateGetActorCallCount()) - }) - - t.Run("parallel", func(t *testing.T) { - api.ResetCallCounts() - dsm := newDealStateMatcher(state.NewStatePredicates(api)) - matcher := dsm.matcher(ctx, abi.DealID(1)) - - // Call matcher with lots of go-routines in parallel - var eg errgroup.Group - res := make([]struct { - ok bool - stateChange events.StateChange - }, 20) - for i := 0; i < len(res); i++ { - i := i - eg.Go(func() error { - ok, stateChange, err := matcher(ts1, ts2) - res[i].ok = ok - res[i].stateChange = stateChange - return err - }) - } - err := eg.Wait() - require.NoError(t, err) - - // All go-routines should have got the same (cached) result - for i := 1; i < len(res); i++ { - require.Equal(t, res[i].ok, res[i-1].ok) - require.Equal(t, res[i].stateChange, res[i-1].stateChange) - } - - // Only one go-routine should have called StateGetActor - // (once for each tipset) - require.Equal(t, 2, api.StateGetActorCallCount()) - }) -} - -func createMarketState(ctx context.Context, t *testing.T, store adt2.Store, deals map[abi.DealID]*market2.DealState) cid.Cid { - dealRootCid := test.CreateDealAMT(ctx, t, store, deals) - state := test.CreateEmptyMarketState(t, store) - state.States = dealRootCid - - stateC, err := store.Put(ctx, state) - require.NoError(t, err) - return stateC -} diff --git a/markets/storageadapter/ondealsectorcommitted.go b/markets/storageadapter/ondealsectorcommitted.go deleted file mode 100644 index e922da0cf..000000000 --- a/markets/storageadapter/ondealsectorcommitted.go +++ /dev/null @@ -1,399 +0,0 @@ -package storageadapter - -import ( - "bytes" - "context" - "sync" - - "github.com/ipfs/go-cid" - "golang.org/x/xerrors" - - "github.com/filecoin-project/boost-gfm/storagemarket" - "github.com/filecoin-project/go-address" - "github.com/filecoin-project/go-bitfield" - "github.com/filecoin-project/go-state-types/abi" - "github.com/filecoin-project/go-state-types/builtin" - "github.com/filecoin-project/go-state-types/builtin/v8/miner" - "github.com/filecoin-project/go-state-types/builtin/v9/market" - - "github.com/filecoin-project/lotus/build" - lminer "github.com/filecoin-project/lotus/chain/actors/builtin/miner" - "github.com/filecoin-project/lotus/chain/events" - "github.com/filecoin-project/lotus/chain/types" - pipeline "github.com/filecoin-project/lotus/storage/pipeline" -) - -type eventsCalledAPI interface { - Called(ctx context.Context, check events.CheckFunc, msgHnd events.MsgHandler, rev events.RevertHandler, confidence int, timeout abi.ChainEpoch, mf events.MsgMatchFunc) error -} - -type dealInfoAPI interface { - GetCurrentDealInfo(ctx context.Context, tsk types.TipSetKey, proposal *market.DealProposal, publishCid cid.Cid) (pipeline.CurrentDealInfo, error) -} - -type diffPreCommitsAPI interface { - diffPreCommits(ctx context.Context, actor address.Address, pre, cur types.TipSetKey) (*lminer.PreCommitChanges, error) -} - -type SectorCommittedManager struct { - ev eventsCalledAPI - dealInfo dealInfoAPI - dpc diffPreCommitsAPI -} - -func NewSectorCommittedManager(ev eventsCalledAPI, tskAPI pipeline.CurrentDealInfoAPI, dpcAPI diffPreCommitsAPI) *SectorCommittedManager { - dim := &pipeline.CurrentDealInfoManager{ - CDAPI: tskAPI, - } - return newSectorCommittedManager(ev, dim, dpcAPI) -} - -func newSectorCommittedManager(ev eventsCalledAPI, dealInfo dealInfoAPI, dpcAPI diffPreCommitsAPI) *SectorCommittedManager { - return &SectorCommittedManager{ - ev: ev, - dealInfo: dealInfo, - dpc: dpcAPI, - } -} - -func (mgr *SectorCommittedManager) OnDealSectorPreCommitted(ctx context.Context, provider address.Address, proposal market.DealProposal, publishCid cid.Cid, callback storagemarket.DealSectorPreCommittedCallback) error { - // Ensure callback is only called once - var once sync.Once - cb := func(sectorNumber abi.SectorNumber, isActive bool, err error) { - once.Do(func() { - callback(sectorNumber, isActive, err) - }) - } - - // First check if the deal is already active, and if so, bail out - checkFunc := func(ctx context.Context, ts *types.TipSet) (done bool, more bool, err error) { - dealInfo, isActive, err := mgr.checkIfDealAlreadyActive(ctx, ts, &proposal, publishCid) - if err != nil { - // Note: the error returned from here will end up being returned - // from OnDealSectorPreCommitted so no need to call the callback - // with the error - return false, false, xerrors.Errorf("failed to check deal activity: %w", err) - } - - if isActive { - // Deal is already active, bail out - cb(0, true, nil) - return true, false, nil - } - - // Check that precommits which landed between when the deal was published - // and now don't already contain the deal we care about. - // (this can happen when the precommit lands vary quickly (in tests), or - // when the client node was down after the deal was published, and when - // the precommit containing it landed on chain) - - diff, err := mgr.dpc.diffPreCommits(ctx, provider, dealInfo.PublishMsgTipSet, ts.Key()) - if err != nil { - return false, false, xerrors.Errorf("failed to diff precommits: %w", err) - } - - for _, info := range diff.Added { - for _, d := range info.Info.DealIDs { - if d == dealInfo.DealID { - cb(info.Info.SectorNumber, false, nil) - return true, false, nil - } - } - } - - // Not yet active, start matching against incoming messages - return false, true, nil - } - - // Watch for a pre-commit message to the provider. - matchEvent := func(msg *types.Message) (bool, error) { - matched := msg.To == provider && (msg.Method == builtin.MethodsMiner.PreCommitSector || msg.Method == builtin.MethodsMiner.PreCommitSectorBatch || msg.Method == builtin.MethodsMiner.ProveReplicaUpdates) - return matched, nil - } - - // The deal must be accepted by the deal proposal start epoch, so timeout - // if the chain reaches that epoch - timeoutEpoch := proposal.StartEpoch + 1 - - // Check if the message params included the deal ID we're looking for. - called := func(msg *types.Message, rec *types.MessageReceipt, ts *types.TipSet, curH abi.ChainEpoch) (more bool, err error) { - defer func() { - if err != nil { - cb(0, false, xerrors.Errorf("handling applied event: %w", err)) - } - }() - - // If the deal hasn't been activated by the proposed start epoch, the - // deal will timeout (when msg == nil it means the timeout epoch was reached) - if msg == nil { - err = xerrors.Errorf("deal with piece CID %s was not activated by proposed deal start epoch %d", proposal.PieceCID, proposal.StartEpoch) - return false, err - } - - // Ignore the pre-commit message if it was not executed successfully - if rec.ExitCode != 0 { - return true, nil - } - - // When there is a reorg, the deal ID may change, so get the - // current deal ID from the publish message CID - res, err := mgr.dealInfo.GetCurrentDealInfo(ctx, ts.Key(), &proposal, publishCid) - if err != nil { - return false, xerrors.Errorf("failed to get dealinfo: %w", err) - } - - // If this is a replica update method that succeeded the deal is active - if msg.Method == builtin.MethodsMiner.ProveReplicaUpdates { - sn, err := dealSectorInReplicaUpdateSuccess(msg, rec, res) - if err != nil { - return false, err - } - if sn != nil { - cb(*sn, true, nil) - return false, nil - } - // Didn't find the deal ID in this message, so keep looking - return true, nil - } - - // Extract the message parameters - sn, err := dealSectorInPreCommitMsg(msg, res) - if err != nil { - return false, xerrors.Errorf("failed to extract message params: %w", err) - } - - if sn != nil { - cb(*sn, false, nil) - } - - // Didn't find the deal ID in this message, so keep looking - return true, nil - } - - revert := func(ctx context.Context, ts *types.TipSet) error { - log.Warn("deal pre-commit reverted; TODO: actually handle this!") - // TODO: Just go back to DealSealing? - return nil - } - - if err := mgr.ev.Called(ctx, checkFunc, called, revert, int(build.MessageConfidence+1), timeoutEpoch, matchEvent); err != nil { - return xerrors.Errorf("failed to set up called handler: %w", err) - } - - return nil -} - -func (mgr *SectorCommittedManager) OnDealSectorCommitted(ctx context.Context, provider address.Address, sectorNumber abi.SectorNumber, proposal market.DealProposal, publishCid cid.Cid, callback storagemarket.DealSectorCommittedCallback) error { - // Ensure callback is only called once - var once sync.Once - cb := func(err error) { - once.Do(func() { - callback(err) - }) - } - - // First check if the deal is already active, and if so, bail out - checkFunc := func(ctx context.Context, ts *types.TipSet) (done bool, more bool, err error) { - _, isActive, err := mgr.checkIfDealAlreadyActive(ctx, ts, &proposal, publishCid) - if err != nil { - // Note: the error returned from here will end up being returned - // from OnDealSectorCommitted so no need to call the callback - // with the error - return false, false, err - } - - if isActive { - // Deal is already active, bail out - cb(nil) - return true, false, nil - } - - // Not yet active, start matching against incoming messages - return false, true, nil - } - - // Match a prove-commit sent to the provider with the given sector number - matchEvent := func(msg *types.Message) (matched bool, err error) { - if msg.To != provider { - return false, nil - } - - return sectorInCommitMsg(msg, sectorNumber) - } - - // The deal must be accepted by the deal proposal start epoch, so timeout - // if the chain reaches that epoch - timeoutEpoch := proposal.StartEpoch + 1 - - called := func(msg *types.Message, rec *types.MessageReceipt, ts *types.TipSet, curH abi.ChainEpoch) (more bool, err error) { - defer func() { - if err != nil { - cb(xerrors.Errorf("handling applied event: %w", err)) - } - }() - - // If the deal hasn't been activated by the proposed start epoch, the - // deal will timeout (when msg == nil it means the timeout epoch was reached) - if msg == nil { - err := xerrors.Errorf("deal with piece CID %s was not activated by proposed deal start epoch %d", proposal.PieceCID, proposal.StartEpoch) - return false, err - } - - // Ignore the prove-commit message if it was not executed successfully - if rec.ExitCode != 0 { - return true, nil - } - - // Get the deal info - res, err := mgr.dealInfo.GetCurrentDealInfo(ctx, ts.Key(), &proposal, publishCid) - if err != nil { - return false, xerrors.Errorf("failed to look up deal on chain: %w", err) - } - - // Make sure the deal is active - if res.MarketDeal.State.SectorStartEpoch < 1 { - return false, xerrors.Errorf("deal wasn't active: deal=%d, parentState=%s, h=%d", res.DealID, ts.ParentState(), ts.Height()) - } - - log.Infof("Storage deal %d activated at epoch %d", res.DealID, res.MarketDeal.State.SectorStartEpoch) - - cb(nil) - - return false, nil - } - - revert := func(ctx context.Context, ts *types.TipSet) error { - log.Warn("deal activation reverted; TODO: actually handle this!") - // TODO: Just go back to DealSealing? - return nil - } - - if err := mgr.ev.Called(ctx, checkFunc, called, revert, int(build.MessageConfidence+1), timeoutEpoch, matchEvent); err != nil { - return xerrors.Errorf("failed to set up called handler: %w", err) - } - - return nil -} - -func dealSectorInReplicaUpdateSuccess(msg *types.Message, rec *types.MessageReceipt, res pipeline.CurrentDealInfo) (*abi.SectorNumber, error) { - var params miner.ProveReplicaUpdatesParams - if err := params.UnmarshalCBOR(bytes.NewReader(msg.Params)); err != nil { - return nil, xerrors.Errorf("unmarshal prove replica update: %w", err) - } - - var seekUpdate miner.ReplicaUpdate - var found bool - for _, update := range params.Updates { - for _, did := range update.Deals { - if did == res.DealID { - seekUpdate = update - found = true - break - } - } - } - if !found { - return nil, nil - } - - // check that this update passed validation steps - var successBf bitfield.BitField - if err := successBf.UnmarshalCBOR(bytes.NewReader(rec.Return)); err != nil { - return nil, xerrors.Errorf("unmarshal return value: %w", err) - } - success, err := successBf.IsSet(uint64(seekUpdate.SectorID)) - if err != nil { - return nil, xerrors.Errorf("failed to check success of replica update: %w", err) - } - if !success { - return nil, xerrors.Errorf("replica update %d failed", seekUpdate.SectorID) - } - return &seekUpdate.SectorID, nil -} - -// dealSectorInPreCommitMsg tries to find a sector containing the specified deal -func dealSectorInPreCommitMsg(msg *types.Message, res pipeline.CurrentDealInfo) (*abi.SectorNumber, error) { - switch msg.Method { - case builtin.MethodsMiner.PreCommitSector: - var params miner.SectorPreCommitInfo - if err := params.UnmarshalCBOR(bytes.NewReader(msg.Params)); err != nil { - return nil, xerrors.Errorf("unmarshal pre commit: %w", err) - } - - // Check through the deal IDs associated with this message - for _, did := range params.DealIDs { - if did == res.DealID { - // Found the deal ID in this message. Callback with the sector ID. - return ¶ms.SectorNumber, nil - } - } - case builtin.MethodsMiner.PreCommitSectorBatch: - var params miner.PreCommitSectorBatchParams - if err := params.UnmarshalCBOR(bytes.NewReader(msg.Params)); err != nil { - return nil, xerrors.Errorf("unmarshal pre commit: %w", err) - } - - for _, precommit := range params.Sectors { - // Check through the deal IDs associated with this message - for _, did := range precommit.DealIDs { - if did == res.DealID { - // Found the deal ID in this message. Callback with the sector ID. - return &precommit.SectorNumber, nil - } - } - } - default: - return nil, xerrors.Errorf("unexpected method %d", msg.Method) - } - - return nil, nil -} - -// sectorInCommitMsg checks if the provided message commits specified sector -func sectorInCommitMsg(msg *types.Message, sectorNumber abi.SectorNumber) (bool, error) { - switch msg.Method { - case builtin.MethodsMiner.ProveCommitSector: - var params miner.ProveCommitSectorParams - if err := params.UnmarshalCBOR(bytes.NewReader(msg.Params)); err != nil { - return false, xerrors.Errorf("failed to unmarshal prove commit sector params: %w", err) - } - - return params.SectorNumber == sectorNumber, nil - - case builtin.MethodsMiner.ProveCommitAggregate: - var params miner.ProveCommitAggregateParams - if err := params.UnmarshalCBOR(bytes.NewReader(msg.Params)); err != nil { - return false, xerrors.Errorf("failed to unmarshal prove commit sector params: %w", err) - } - - set, err := params.SectorNumbers.IsSet(uint64(sectorNumber)) - if err != nil { - return false, xerrors.Errorf("checking if sectorNumber is set in commit aggregate message: %w", err) - } - - return set, nil - - default: - return false, nil - } -} - -func (mgr *SectorCommittedManager) checkIfDealAlreadyActive(ctx context.Context, ts *types.TipSet, proposal *market.DealProposal, publishCid cid.Cid) (pipeline.CurrentDealInfo, bool, error) { - res, err := mgr.dealInfo.GetCurrentDealInfo(ctx, ts.Key(), proposal, publishCid) - if err != nil { - // TODO: This may be fine for some errors - return res, false, xerrors.Errorf("failed to look up deal on chain: %w", err) - } - - // Sector was slashed - if res.MarketDeal.State.SlashEpoch > 0 { - return res, false, xerrors.Errorf("deal %d was slashed at epoch %d", res.DealID, res.MarketDeal.State.SlashEpoch) - } - - // Sector with deal is already active - if res.MarketDeal.State.SectorStartEpoch > 0 { - return res, true, nil - } - - return res, false, nil -} diff --git a/markets/storageadapter/ondealsectorcommitted_test.go b/markets/storageadapter/ondealsectorcommitted_test.go deleted file mode 100644 index 6edefc2aa..000000000 --- a/markets/storageadapter/ondealsectorcommitted_test.go +++ /dev/null @@ -1,581 +0,0 @@ -// stm: #unit -package storageadapter - -import ( - "bytes" - "context" - "errors" - "fmt" - "math/rand" - "testing" - "time" - - "github.com/filecoin-project/go-address" - "github.com/filecoin-project/go-state-types/abi" - "github.com/filecoin-project/go-state-types/builtin" - markettypes "github.com/filecoin-project/go-state-types/builtin/v9/market" - minertypes "github.com/filecoin-project/go-state-types/builtin/v9/miner" - "github.com/filecoin-project/go-state-types/cbor" - "github.com/filecoin-project/lotus/api" - "github.com/filecoin-project/lotus/chain/actors/builtin/market" - "github.com/filecoin-project/lotus/chain/actors/builtin/miner" - "github.com/filecoin-project/lotus/chain/events" - test "github.com/filecoin-project/lotus/chain/events/state/mock" - "github.com/filecoin-project/lotus/chain/types" - pipeline "github.com/filecoin-project/lotus/storage/pipeline" - tutils "github.com/filecoin-project/specs-actors/v2/support/testing" - blocks "github.com/ipfs/go-block-format" - "github.com/ipfs/go-cid" - "github.com/stretchr/testify/require" - "golang.org/x/xerrors" -) - -func TestOnDealSectorPreCommitted(t *testing.T) { - label, err := markettypes.NewLabelFromString("success") - require.NoError(t, err) - - provider := address.TestAddress - ctx := context.Background() - publishCid := generateCids(1)[0] - sealedCid := generateCids(1)[0] - pieceCid := generateCids(1)[0] - dealID := abi.DealID(rand.Uint64()) - sectorNumber := abi.SectorNumber(rand.Uint64()) - proposal := market.DealProposal{ - PieceCID: pieceCid, - PieceSize: abi.PaddedPieceSize(rand.Uint64()), - Client: tutils.NewActorAddr(t, "client"), - Provider: tutils.NewActorAddr(t, "provider"), - StoragePricePerEpoch: abi.NewTokenAmount(1), - ProviderCollateral: abi.NewTokenAmount(1), - ClientCollateral: abi.NewTokenAmount(1), - Label: label, - } - unfinishedDeal := &api.MarketDeal{ - Proposal: proposal, - State: market.DealState{ - SectorStartEpoch: -1, - LastUpdatedEpoch: 2, - }, - } - activeDeal := &api.MarketDeal{ - Proposal: proposal, - State: market.DealState{ - SectorStartEpoch: 1, - LastUpdatedEpoch: 2, - }, - } - slashedDeal := &api.MarketDeal{ - Proposal: proposal, - State: market.DealState{ - SectorStartEpoch: 1, - LastUpdatedEpoch: 2, - SlashEpoch: 2, - }, - } - type testCase struct { - currentDealInfo pipeline.CurrentDealInfo - currentDealInfoErr error - currentDealInfoErr2 error - preCommitDiff *miner.PreCommitChanges - matchStates []matchState - dealStartEpochTimeout bool - expectedCBCallCount uint64 - expectedCBSectorNumber abi.SectorNumber - expectedCBIsActive bool - expectedCBError error - expectedError error - } - testCases := map[string]testCase{ - "normal sequence": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: unfinishedDeal, - }, - matchStates: []matchState{ - { - msg: makeMessage(t, provider, builtin.MethodsMiner.PreCommitSector, &minertypes.PreCommitSectorParams{ - SectorNumber: sectorNumber, - SealedCID: sealedCid, - DealIDs: []abi.DealID{dealID}, - }), - }, - }, - expectedCBCallCount: 1, - expectedCBIsActive: false, - expectedCBSectorNumber: sectorNumber, - }, - "ignores unsuccessful pre-commit message": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: unfinishedDeal, - }, - matchStates: []matchState{ - { - msg: makeMessage(t, provider, builtin.MethodsMiner.PreCommitSector, &minertypes.PreCommitSectorParams{ - SectorNumber: sectorNumber, - SealedCID: sealedCid, - DealIDs: []abi.DealID{dealID}, - }), - // non-zero exit code indicates unsuccessful pre-commit message - receipt: &types.MessageReceipt{ExitCode: 1}, - }, - }, - expectedCBCallCount: 0, - }, - "deal already pre-committed": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: unfinishedDeal, - }, - preCommitDiff: &miner.PreCommitChanges{ - Added: []minertypes.SectorPreCommitOnChainInfo{{ - Info: minertypes.SectorPreCommitInfo{ - SectorNumber: sectorNumber, - DealIDs: []abi.DealID{dealID}, - }, - }}, - }, - expectedCBCallCount: 1, - expectedCBIsActive: false, - expectedCBSectorNumber: sectorNumber, - }, - "error getting current deal info in check func": { - currentDealInfoErr: errors.New("something went wrong"), - expectedCBCallCount: 0, - expectedError: xerrors.Errorf("failed to set up called handler: failed to check deal activity: failed to look up deal on chain: something went wrong"), - }, - "sector already active": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: activeDeal, - }, - expectedCBCallCount: 1, - expectedCBIsActive: true, - }, - "sector was slashed": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: slashedDeal, - PublishMsgTipSet: types.EmptyTSK, - }, - expectedCBCallCount: 0, - expectedError: xerrors.Errorf("failed to set up called handler: failed to check deal activity: deal %d was slashed at epoch %d", dealID, slashedDeal.State.SlashEpoch), - }, - "error getting current deal info in called func": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: unfinishedDeal, - }, - currentDealInfoErr2: errors.New("something went wrong"), - matchStates: []matchState{ - { - msg: makeMessage(t, provider, builtin.MethodsMiner.PreCommitSector, &minertypes.PreCommitSectorParams{ - SectorNumber: sectorNumber, - SealedCID: sealedCid, - DealIDs: []abi.DealID{dealID}, - }), - }, - }, - expectedCBCallCount: 1, - expectedCBError: errors.New("handling applied event: failed to get dealinfo: something went wrong"), - }, - "proposed deal epoch timeout": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: activeDeal, - }, - dealStartEpochTimeout: true, - expectedCBCallCount: 1, - expectedCBError: xerrors.Errorf("handling applied event: deal with piece CID %s was not activated by proposed deal start epoch 0", unfinishedDeal.Proposal.PieceCID), - }, - } - runTestCase := func(testCase string, data testCase) { - t.Run(testCase, func(t *testing.T) { - checkTs, err := test.MockTipset(provider, rand.Uint64()) - require.NoError(t, err) - matchMessages := make([]matchMessage, len(data.matchStates)) - for i, ms := range data.matchStates { - matchTs, err := test.MockTipset(provider, rand.Uint64()) - require.NoError(t, err) - matchMessages[i] = matchMessage{ - curH: 5, - msg: ms.msg, - msgReceipt: ms.receipt, - ts: matchTs, - } - } - eventsAPI := &fakeEvents{ - Ctx: ctx, - CheckTs: checkTs, - MatchMessages: matchMessages, - DealStartEpochTimeout: data.dealStartEpochTimeout, - } - cbCallCount := uint64(0) - var cbSectorNumber abi.SectorNumber - var cbIsActive bool - var cbError error - cb := func(secNum abi.SectorNumber, isActive bool, err error) { - cbCallCount++ - cbSectorNumber = secNum - cbIsActive = isActive - cbError = err - } - - mockPCAPI := &mockPreCommitsAPI{ - PCChanges: data.preCommitDiff, - } - mockDIAPI := &mockDealInfoAPI{ - CurrentDealInfo: data.currentDealInfo, - CurrentDealInfo2: data.currentDealInfo, - Err: data.currentDealInfoErr, - Err2: data.currentDealInfoErr2, - } - scm := newSectorCommittedManager(eventsAPI, mockDIAPI, mockPCAPI) - //stm: @MARKET_ADAPTER_ON_SECTOR_PRE_COMMIT_001 - err = scm.OnDealSectorPreCommitted(ctx, provider, proposal, publishCid, cb) - if data.expectedError == nil { - require.NoError(t, err) - } else { - require.EqualError(t, err, data.expectedError.Error()) - } - require.Equal(t, data.expectedCBSectorNumber, cbSectorNumber) - require.Equal(t, data.expectedCBIsActive, cbIsActive) - require.Equal(t, data.expectedCBCallCount, cbCallCount) - if data.expectedCBError == nil { - require.NoError(t, cbError) - } else { - require.EqualError(t, cbError, data.expectedCBError.Error()) - } - }) - } - for testCase, data := range testCases { - runTestCase(testCase, data) - } -} - -func TestOnDealSectorCommitted(t *testing.T) { - label, err := markettypes.NewLabelFromString("success") - require.NoError(t, err) - - provider := address.TestAddress - publishCid := generateCids(1)[0] - pieceCid := generateCids(1)[0] - dealID := abi.DealID(rand.Uint64()) - sectorNumber := abi.SectorNumber(rand.Uint64()) - proposal := market.DealProposal{ - PieceCID: pieceCid, - PieceSize: abi.PaddedPieceSize(rand.Uint64()), - Client: tutils.NewActorAddr(t, "client"), - Provider: tutils.NewActorAddr(t, "provider"), - StoragePricePerEpoch: abi.NewTokenAmount(1), - ProviderCollateral: abi.NewTokenAmount(1), - ClientCollateral: abi.NewTokenAmount(1), - Label: label, - } - unfinishedDeal := &api.MarketDeal{ - Proposal: proposal, - State: market.DealState{ - SectorStartEpoch: -1, - LastUpdatedEpoch: 2, - }, - } - activeDeal := &api.MarketDeal{ - Proposal: proposal, - State: market.DealState{ - SectorStartEpoch: 1, - LastUpdatedEpoch: 2, - }, - } - slashedDeal := &api.MarketDeal{ - Proposal: proposal, - State: market.DealState{ - SectorStartEpoch: 1, - LastUpdatedEpoch: 2, - SlashEpoch: 2, - }, - } - type testCase struct { - currentDealInfo pipeline.CurrentDealInfo - currentDealInfoErr error - currentDealInfo2 pipeline.CurrentDealInfo - currentDealInfoErr2 error - matchStates []matchState - dealStartEpochTimeout bool - expectedCBCallCount uint64 - expectedCBError error - expectedError error - } - testCases := map[string]testCase{ - "normal sequence": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: unfinishedDeal, - }, - currentDealInfo2: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: activeDeal, - }, - matchStates: []matchState{ - { - msg: makeMessage(t, provider, builtin.MethodsMiner.ProveCommitSector, &minertypes.ProveCommitSectorParams{ - SectorNumber: sectorNumber, - }), - }, - }, - expectedCBCallCount: 1, - }, - "ignores unsuccessful prove-commit message": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: unfinishedDeal, - }, - currentDealInfo2: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: activeDeal, - }, - matchStates: []matchState{ - { - msg: makeMessage(t, provider, builtin.MethodsMiner.ProveCommitSector, &minertypes.ProveCommitSectorParams{ - SectorNumber: sectorNumber, - }), - // Exit-code 1 means the prove-commit was unsuccessful - receipt: &types.MessageReceipt{ExitCode: 1}, - }, - }, - expectedCBCallCount: 0, - }, - "error getting current deal info in check func": { - currentDealInfoErr: errors.New("something went wrong"), - expectedCBCallCount: 0, - expectedError: xerrors.Errorf("failed to set up called handler: failed to look up deal on chain: something went wrong"), - }, - "sector already active": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: activeDeal, - }, - expectedCBCallCount: 1, - }, - "sector was slashed": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: slashedDeal, - }, - expectedCBCallCount: 0, - expectedError: xerrors.Errorf("failed to set up called handler: deal %d was slashed at epoch %d", dealID, slashedDeal.State.SlashEpoch), - }, - "error getting current deal info in called func": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: unfinishedDeal, - }, - currentDealInfoErr2: errors.New("something went wrong"), - matchStates: []matchState{ - { - msg: makeMessage(t, provider, builtin.MethodsMiner.ProveCommitSector, &minertypes.ProveCommitSectorParams{ - SectorNumber: sectorNumber, - }), - }, - }, - expectedCBCallCount: 1, - expectedCBError: xerrors.Errorf("handling applied event: failed to look up deal on chain: something went wrong"), - }, - "proposed deal epoch timeout": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: unfinishedDeal, - }, - dealStartEpochTimeout: true, - expectedCBCallCount: 1, - expectedCBError: xerrors.Errorf("handling applied event: deal with piece CID %s was not activated by proposed deal start epoch 0", unfinishedDeal.Proposal.PieceCID), - }, - "got prove-commit but deal not active": { - currentDealInfo: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: unfinishedDeal, - }, - currentDealInfo2: pipeline.CurrentDealInfo{ - DealID: dealID, - MarketDeal: unfinishedDeal, - }, - matchStates: []matchState{ - { - msg: makeMessage(t, provider, builtin.MethodsMiner.ProveCommitSector, &minertypes.ProveCommitSectorParams{ - SectorNumber: sectorNumber, - }), - }, - }, - expectedCBCallCount: 1, - expectedCBError: xerrors.Errorf("handling applied event: deal wasn't active: deal=%d, parentState=bafkqaaa, h=5", dealID), - }, - } - runTestCase := func(testCase string, data testCase) { - t.Run(testCase, func(t *testing.T) { - ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) - defer cancel() - checkTs, err := test.MockTipset(provider, rand.Uint64()) - require.NoError(t, err) - matchMessages := make([]matchMessage, len(data.matchStates)) - for i, ms := range data.matchStates { - matchTs, err := test.MockTipset(provider, rand.Uint64()) - require.NoError(t, err) - matchMessages[i] = matchMessage{ - curH: 5, - msg: ms.msg, - msgReceipt: ms.receipt, - ts: matchTs, - } - } - eventsAPI := &fakeEvents{ - Ctx: ctx, - CheckTs: checkTs, - MatchMessages: matchMessages, - DealStartEpochTimeout: data.dealStartEpochTimeout, - } - cbCallCount := uint64(0) - var cbError error - cb := func(err error) { - cbCallCount++ - cbError = err - } - mockPCAPI := &mockPreCommitsAPI{} - mockDIAPI := &mockDealInfoAPI{ - CurrentDealInfo: data.currentDealInfo, - CurrentDealInfo2: data.currentDealInfo2, - Err: data.currentDealInfoErr, - Err2: data.currentDealInfoErr2, - } - scm := newSectorCommittedManager(eventsAPI, mockDIAPI, mockPCAPI) - //stm: @MARKET_ADAPTER_ON_SECTOR_COMMIT_001 - err = scm.OnDealSectorCommitted(ctx, provider, sectorNumber, proposal, publishCid, cb) - if data.expectedError == nil { - require.NoError(t, err) - } else { - require.EqualError(t, err, data.expectedError.Error()) - } - require.Equal(t, data.expectedCBCallCount, cbCallCount) - if data.expectedCBError == nil { - require.NoError(t, cbError) - } else { - require.EqualError(t, cbError, data.expectedCBError.Error()) - } - }) - } - for testCase, data := range testCases { - runTestCase(testCase, data) - } -} - -type matchState struct { - msg *types.Message - receipt *types.MessageReceipt -} - -type matchMessage struct { - curH abi.ChainEpoch - msg *types.Message - msgReceipt *types.MessageReceipt - ts *types.TipSet - doesRevert bool -} -type fakeEvents struct { - Ctx context.Context - CheckTs *types.TipSet - MatchMessages []matchMessage - DealStartEpochTimeout bool -} - -func (fe *fakeEvents) Called(ctx context.Context, check events.CheckFunc, msgHnd events.MsgHandler, rev events.RevertHandler, confidence int, timeout abi.ChainEpoch, mf events.MsgMatchFunc) error { - if fe.DealStartEpochTimeout { - msgHnd(nil, nil, nil, 100) // nolint:errcheck - return nil - } - - _, more, err := check(ctx, fe.CheckTs) - if err != nil { - return err - } - if !more { - return nil - } - for _, matchMessage := range fe.MatchMessages { - matched, err := mf(matchMessage.msg) - if err != nil { - return err - } - if matched { - receipt := matchMessage.msgReceipt - if receipt == nil { - receipt = &types.MessageReceipt{ExitCode: 0} - } - more, err := msgHnd(matchMessage.msg, receipt, matchMessage.ts, matchMessage.curH) - if err != nil { - // error is handled through a callback rather than being returned - return nil - } - if matchMessage.doesRevert { - err := rev(ctx, matchMessage.ts) - if err != nil { - return err - } - } - if !more { - return nil - } - } - } - return nil -} - -func makeMessage(t *testing.T, to address.Address, method abi.MethodNum, params cbor.Marshaler) *types.Message { - buf := new(bytes.Buffer) - err := params.MarshalCBOR(buf) - require.NoError(t, err) - return &types.Message{ - To: to, - Method: method, - Params: buf.Bytes(), - } -} - -var seq int - -func generateCids(n int) []cid.Cid { - cids := make([]cid.Cid, 0, n) - for i := 0; i < n; i++ { - c := blocks.NewBlock([]byte(fmt.Sprint(seq))).Cid() - seq++ - cids = append(cids, c) - } - return cids -} - -type mockPreCommitsAPI struct { - PCChanges *miner.PreCommitChanges - Err error -} - -func (m *mockPreCommitsAPI) diffPreCommits(ctx context.Context, actor address.Address, pre, cur types.TipSetKey) (*miner.PreCommitChanges, error) { - pcc := &miner.PreCommitChanges{} - if m.PCChanges != nil { - pcc = m.PCChanges - } - return pcc, m.Err -} - -type mockDealInfoAPI struct { - count int - CurrentDealInfo pipeline.CurrentDealInfo - Err error - CurrentDealInfo2 pipeline.CurrentDealInfo - Err2 error -} - -func (m *mockDealInfoAPI) GetCurrentDealInfo(ctx context.Context, tsk types.TipSetKey, proposal *market.DealProposal, publishCid cid.Cid) (pipeline.CurrentDealInfo, error) { - m.count++ - if m.count == 2 { - return m.CurrentDealInfo2, m.Err2 - } - return m.CurrentDealInfo, m.Err -} diff --git a/markets/storageadapter/provider.go b/markets/storageadapter/provider.go deleted file mode 100644 index a6ddc6b22..000000000 --- a/markets/storageadapter/provider.go +++ /dev/null @@ -1,438 +0,0 @@ -package storageadapter - -// this file implements storagemarket.StorageProviderNode - -import ( - "context" - "errors" - "time" - - "github.com/ipfs/go-cid" - logging "github.com/ipfs/go-log/v2" - "go.uber.org/fx" - "golang.org/x/xerrors" - - "github.com/filecoin-project/boost-gfm/shared" - "github.com/filecoin-project/boost-gfm/storagemarket" - "github.com/filecoin-project/go-address" - "github.com/filecoin-project/go-state-types/abi" - markettypes "github.com/filecoin-project/go-state-types/builtin/v9/market" - "github.com/filecoin-project/go-state-types/crypto" - "github.com/filecoin-project/go-state-types/exitcode" - - "github.com/filecoin-project/boost/markets/utils" - "github.com/filecoin-project/lotus/api" - "github.com/filecoin-project/lotus/api/v1api" - "github.com/filecoin-project/lotus/build" - "github.com/filecoin-project/lotus/chain/actors/builtin/market" - "github.com/filecoin-project/lotus/chain/actors/builtin/miner" - "github.com/filecoin-project/lotus/chain/events" - "github.com/filecoin-project/lotus/chain/events/state" - "github.com/filecoin-project/lotus/chain/types" - "github.com/filecoin-project/lotus/lib/sigs" - "github.com/filecoin-project/lotus/node/config" - "github.com/filecoin-project/lotus/node/modules/helpers" - pipeline "github.com/filecoin-project/lotus/storage/pipeline" - "github.com/filecoin-project/lotus/storage/sectorblocks" -) - -var addPieceRetryWait = 5 * time.Minute -var addPieceRetryTimeout = 6 * time.Hour -var defaultMaxProviderCollateralMultiplier = uint64(2) -var log = logging.Logger("storageadapter") - -type ProviderNodeAdapter struct { - v1api.FullNode - - secb *sectorblocks.SectorBlocks - ev *events.Events - - dealPublisher *DealPublisher - - addBalanceSpec *api.MessageSendSpec - maxDealCollateralMultiplier uint64 - dsMatcher *dealStateMatcher - scMgr *SectorCommittedManager -} - -func NewProviderNodeAdapter(fc *config.MinerFeeConfig, dc *config.DealmakingConfig) func(mctx helpers.MetricsCtx, lc fx.Lifecycle, secb *sectorblocks.SectorBlocks, full v1api.FullNode, dealPublisher *DealPublisher) (storagemarket.StorageProviderNode, error) { - return func(mctx helpers.MetricsCtx, lc fx.Lifecycle, secb *sectorblocks.SectorBlocks, full v1api.FullNode, dealPublisher *DealPublisher) (storagemarket.StorageProviderNode, error) { - ctx := helpers.LifecycleCtx(mctx, lc) - - ev, err := events.NewEvents(ctx, full) - if err != nil { - return nil, err - } - na := &ProviderNodeAdapter{ - FullNode: full, - - secb: secb, - ev: ev, - dealPublisher: dealPublisher, - dsMatcher: newDealStateMatcher(state.NewStatePredicates(state.WrapFastAPI(full))), - } - if fc != nil { - na.addBalanceSpec = &api.MessageSendSpec{MaxFee: abi.TokenAmount(fc.MaxMarketBalanceAddFee)} - } - na.maxDealCollateralMultiplier = defaultMaxProviderCollateralMultiplier - if dc != nil { - na.maxDealCollateralMultiplier = dc.MaxProviderCollateralMultiplier - } - na.scMgr = NewSectorCommittedManager(ev, na, &apiWrapper{api: full}) - - return na, nil - } -} - -func (n *ProviderNodeAdapter) PublishDeals(ctx context.Context, deal storagemarket.MinerDeal) (cid.Cid, error) { - return n.dealPublisher.Publish(ctx, deal.ClientDealProposal) -} - -func (n *ProviderNodeAdapter) OnDealComplete(ctx context.Context, deal storagemarket.MinerDeal, pieceSize abi.UnpaddedPieceSize, pieceData shared.ReadSeekStarter) (*storagemarket.PackingResult, error) { - if deal.PublishCid == nil { - return nil, xerrors.Errorf("deal.PublishCid can't be nil") - } - - sdInfo := api.PieceDealInfo{ - DealID: deal.DealID, - DealProposal: &deal.Proposal, - PublishCid: deal.PublishCid, - DealSchedule: api.DealSchedule{ - StartEpoch: deal.ClientDealProposal.Proposal.StartEpoch, - EndEpoch: deal.ClientDealProposal.Proposal.EndEpoch, - }, - KeepUnsealed: deal.FastRetrieval, - } - - // Attempt to add the piece to the sector - p, offset, err := n.secb.AddPiece(ctx, pieceSize, pieceData, sdInfo) - curTime := build.Clock.Now() - for build.Clock.Since(curTime) < addPieceRetryTimeout { - // Check if there was an error because of too many sectors being sealed - if !errors.Is(err, pipeline.ErrTooManySectorsSealing) { - if err != nil { - log.Errorf("failed to addPiece for deal %d, err: %v", deal.DealID, err) - } - - // There was either a fatal error or no error. In either case - // don't retry AddPiece - break - } - - // The piece could not be added to the sector because there are too - // many sectors being sealed, back-off for a while before trying again - select { - case <-build.Clock.After(addPieceRetryWait): - // Reset the reader to the start - err = pieceData.SeekStart() - if err != nil { - return nil, xerrors.Errorf("failed to reset piece reader to start before retrying AddPiece for deal %d: %w", deal.DealID, err) - } - - // Attempt to add the piece again - p, offset, err = n.secb.AddPiece(ctx, pieceSize, pieceData, sdInfo) - case <-ctx.Done(): - return nil, xerrors.New("context expired while waiting to retry AddPiece") - } - } - - if err != nil { - return nil, xerrors.Errorf("AddPiece failed: %s", err) - } - log.Warnf("New Deal: deal %d", deal.DealID) - - return &storagemarket.PackingResult{ - SectorNumber: p, - Offset: offset, - Size: pieceSize.Padded(), - }, nil -} - -func (n *ProviderNodeAdapter) VerifySignature(ctx context.Context, sig crypto.Signature, addr address.Address, input []byte, encodedTs shared.TipSetToken) (bool, error) { - addr, err := n.StateAccountKey(ctx, addr, types.EmptyTSK) - if err != nil { - return false, err - } - - err = sigs.Verify(&sig, addr, input) - return err == nil, err -} - -func (n *ProviderNodeAdapter) GetMinerWorkerAddress(ctx context.Context, maddr address.Address, tok shared.TipSetToken) (address.Address, error) { - tsk, err := types.TipSetKeyFromBytes(tok) - if err != nil { - return address.Undef, err - } - - mi, err := n.StateMinerInfo(ctx, maddr, tsk) - if err != nil { - return address.Address{}, err - } - return mi.Worker, nil -} - -func (n *ProviderNodeAdapter) GetProofType(ctx context.Context, maddr address.Address, tok shared.TipSetToken) (abi.RegisteredSealProof, error) { - tsk, err := types.TipSetKeyFromBytes(tok) - if err != nil { - return 0, err - } - - mi, err := n.StateMinerInfo(ctx, maddr, tsk) - if err != nil { - return 0, err - } - - nver, err := n.StateNetworkVersion(ctx, tsk) - if err != nil { - return 0, err - } - - return miner.PreferredSealProofTypeFromWindowPoStType(nver, mi.WindowPoStProofType, false) -} - -func (n *ProviderNodeAdapter) SignBytes(ctx context.Context, signer address.Address, b []byte) (*crypto.Signature, error) { - signer, err := n.StateAccountKey(ctx, signer, types.EmptyTSK) - if err != nil { - return nil, err - } - - localSignature, err := n.WalletSign(ctx, signer, b) - if err != nil { - return nil, err - } - return localSignature, nil -} - -func (n *ProviderNodeAdapter) ReserveFunds(ctx context.Context, wallet, addr address.Address, amt abi.TokenAmount) (cid.Cid, error) { - return n.MarketReserveFunds(ctx, wallet, addr, amt) -} - -func (n *ProviderNodeAdapter) ReleaseFunds(ctx context.Context, addr address.Address, amt abi.TokenAmount) error { - return n.MarketReleaseFunds(ctx, addr, amt) -} - -// Adds funds with the StorageMinerActor for a storage participant. Used by both providers and clients. -func (n *ProviderNodeAdapter) AddFunds(ctx context.Context, addr address.Address, amount abi.TokenAmount) (cid.Cid, error) { - // (Provider Node API) - smsg, err := n.MpoolPushMessage(ctx, &types.Message{ - To: market.Address, - From: addr, - Value: amount, - Method: market.Methods.AddBalance, - }, n.addBalanceSpec) - if err != nil { - return cid.Undef, err - } - - return smsg.Cid(), nil -} - -func (n *ProviderNodeAdapter) GetBalance(ctx context.Context, addr address.Address, encodedTs shared.TipSetToken) (storagemarket.Balance, error) { - tsk, err := types.TipSetKeyFromBytes(encodedTs) - if err != nil { - return storagemarket.Balance{}, err - } - - bal, err := n.StateMarketBalance(ctx, addr, tsk) - if err != nil { - return storagemarket.Balance{}, err - } - - return utils.ToSharedBalance(bal), nil -} - -// TODO: why doesnt this method take in a sector ID? -func (n *ProviderNodeAdapter) LocatePieceForDealWithinSector(ctx context.Context, dealID abi.DealID, encodedTs shared.TipSetToken) (sectorID abi.SectorNumber, offset abi.PaddedPieceSize, length abi.PaddedPieceSize, err error) { - refs, err := n.secb.GetRefs(ctx, dealID) - if err != nil { - return 0, 0, 0, err - } - if len(refs) == 0 { - return 0, 0, 0, xerrors.New("no sector information for deal ID") - } - - // TODO: better strategy (e.g. look for already unsealed) - var best api.SealedRef - var bestSi api.SectorInfo - for _, r := range refs { - si, err := n.secb.SectorBuilder.SectorsStatus(ctx, r.SectorID, false) - if err != nil { - return 0, 0, 0, xerrors.Errorf("getting sector info: %w", err) - } - if si.State == api.SectorState(pipeline.Proving) { - best = r - bestSi = si - break - } - } - if bestSi.State == api.SectorState(pipeline.UndefinedSectorState) { - return 0, 0, 0, xerrors.New("no sealed sector found") - } - return best.SectorID, best.Offset, best.Size.Padded(), nil -} - -func (n *ProviderNodeAdapter) DealProviderCollateralBounds(ctx context.Context, size abi.PaddedPieceSize, isVerified bool) (abi.TokenAmount, abi.TokenAmount, error) { - bounds, err := n.StateDealProviderCollateralBounds(ctx, size, isVerified, types.EmptyTSK) - if err != nil { - return abi.TokenAmount{}, abi.TokenAmount{}, err - } - - // The maximum amount of collateral that the provider will put into escrow - // for a deal is calculated as a multiple of the minimum bounded amount - max := types.BigMul(bounds.Min, types.NewInt(n.maxDealCollateralMultiplier)) - - return bounds.Min, max, nil -} - -// TODO: Remove dealID parameter, change publishCid to be cid.Cid (instead of pointer) -func (n *ProviderNodeAdapter) OnDealSectorPreCommitted(ctx context.Context, provider address.Address, dealID abi.DealID, proposal markettypes.DealProposal, publishCid *cid.Cid, cb storagemarket.DealSectorPreCommittedCallback) error { - return n.scMgr.OnDealSectorPreCommitted(ctx, provider, proposal, *publishCid, cb) -} - -// TODO: Remove dealID parameter, change publishCid to be cid.Cid (instead of pointer) -func (n *ProviderNodeAdapter) OnDealSectorCommitted(ctx context.Context, provider address.Address, dealID abi.DealID, sectorNumber abi.SectorNumber, proposal markettypes.DealProposal, publishCid *cid.Cid, cb storagemarket.DealSectorCommittedCallback) error { - return n.scMgr.OnDealSectorCommitted(ctx, provider, sectorNumber, proposal, *publishCid, cb) -} - -func (n *ProviderNodeAdapter) GetChainHead(ctx context.Context) (shared.TipSetToken, abi.ChainEpoch, error) { - head, err := n.ChainHead(ctx) - if err != nil { - return nil, 0, err - } - - return head.Key().Bytes(), head.Height(), nil -} - -func (n *ProviderNodeAdapter) WaitForMessage(ctx context.Context, mcid cid.Cid, cb func(code exitcode.ExitCode, bytes []byte, finalCid cid.Cid, err error) error) error { - receipt, err := n.StateWaitMsg(ctx, mcid, 2*build.MessageConfidence, api.LookbackNoLimit, true) - if err != nil { - return cb(0, nil, cid.Undef, err) - } - return cb(receipt.Receipt.ExitCode, receipt.Receipt.Return, receipt.Message, nil) -} - -func (n *ProviderNodeAdapter) WaitForPublishDeals(ctx context.Context, publishCid cid.Cid, proposal markettypes.DealProposal) (*storagemarket.PublishDealsWaitResult, error) { - // Wait for deal to be published (plus additional time for confidence) - receipt, err := n.StateWaitMsg(ctx, publishCid, 2*build.MessageConfidence, api.LookbackNoLimit, true) - if err != nil { - return nil, xerrors.Errorf("WaitForPublishDeals errored: %w", err) - } - if receipt.Receipt.ExitCode != exitcode.Ok { - return nil, xerrors.Errorf("WaitForPublishDeals exit code: %s", receipt.Receipt.ExitCode) - } - - // The deal ID may have changed since publish if there was a reorg, so - // get the current deal ID - head, err := n.ChainHead(ctx) - if err != nil { - return nil, xerrors.Errorf("WaitForPublishDeals failed to get chain head: %w", err) - } - - res, err := n.scMgr.dealInfo.GetCurrentDealInfo(ctx, head.Key(), &proposal, publishCid) - if err != nil { - return nil, xerrors.Errorf("WaitForPublishDeals getting deal info errored: %w", err) - } - - return &storagemarket.PublishDealsWaitResult{DealID: res.DealID, FinalCid: receipt.Message}, nil -} - -func (n *ProviderNodeAdapter) GetDataCap(ctx context.Context, addr address.Address, encodedTs shared.TipSetToken) (*abi.StoragePower, error) { - tsk, err := types.TipSetKeyFromBytes(encodedTs) - if err != nil { - return nil, err - } - - sp, err := n.StateVerifiedClientStatus(ctx, addr, tsk) - return sp, err -} - -func (n *ProviderNodeAdapter) OnDealExpiredOrSlashed(ctx context.Context, dealID abi.DealID, onDealExpired storagemarket.DealExpiredCallback, onDealSlashed storagemarket.DealSlashedCallback) error { - head, err := n.ChainHead(ctx) - if err != nil { - return xerrors.Errorf("client: failed to get chain head: %w", err) - } - - sd, err := n.StateMarketStorageDeal(ctx, dealID, head.Key()) - if err != nil { - return xerrors.Errorf("client: failed to look up deal %d on chain: %w", dealID, err) - } - - // Called immediately to check if the deal has already expired or been slashed - checkFunc := func(ctx context.Context, ts *types.TipSet) (done bool, more bool, err error) { - if ts == nil { - // keep listening for events - return false, true, nil - } - - // Check if the deal has already expired - if sd.Proposal.EndEpoch <= ts.Height() { - onDealExpired(nil) - return true, false, nil - } - - // If there is no deal assume it's already been slashed - if sd.State.SectorStartEpoch < 0 { - onDealSlashed(ts.Height(), nil) - return true, false, nil - } - - // No events have occurred yet, so return - // done: false, more: true (keep listening for events) - return false, true, nil - } - - // Called when there was a match against the state change we're looking for - // and the chain has advanced to the confidence height - stateChanged := func(ts *types.TipSet, ts2 *types.TipSet, states events.StateChange, h abi.ChainEpoch) (more bool, err error) { - // Check if the deal has already expired - if ts2 == nil || sd.Proposal.EndEpoch <= ts2.Height() { - onDealExpired(nil) - return false, nil - } - - // Timeout waiting for state change - if states == nil { - log.Error("timed out waiting for deal expiry") - return false, nil - } - - changedDeals, ok := states.(state.ChangedDeals) - if !ok { - panic("Expected state.ChangedDeals") - } - - deal, ok := changedDeals[dealID] - if !ok { - // No change to deal - return true, nil - } - - // Deal was slashed - if deal.To == nil { - onDealSlashed(ts2.Height(), nil) - return false, nil - } - - return true, nil - } - - // Called when there was a chain reorg and the state change was reverted - revert := func(ctx context.Context, ts *types.TipSet) error { - // TODO: Is it ok to just ignore this? - log.Warn("deal state reverted; TODO: actually handle this!") - return nil - } - - // Watch for state changes to the deal - match := n.dsMatcher.matcher(ctx, dealID) - - // Wait until after the end epoch for the deal and then timeout - timeout := (sd.Proposal.EndEpoch - head.Height()) + 1 - if err := n.ev.StateChanged(checkFunc, stateChanged, revert, int(build.MessageConfidence)+1, timeout, match); err != nil { - return xerrors.Errorf("failed to set up state changed handler: %w", err) - } - - return nil -} - -var _ storagemarket.StorageProviderNode = &ProviderNodeAdapter{} diff --git a/markets/utils/converters.go b/markets/utils/converters.go index 3ff0b0475..c6fc899b7 100644 --- a/markets/utils/converters.go +++ b/markets/utils/converters.go @@ -1,10 +1,10 @@ package utils import ( + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" "github.com/libp2p/go-libp2p/core/peer" "github.com/multiformats/go-multiaddr" - "github.com/filecoin-project/boost-gfm/storagemarket" "github.com/filecoin-project/go-address" "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/go-state-types/big" @@ -12,17 +12,17 @@ import ( "github.com/filecoin-project/lotus/api" ) -func NewStorageProviderInfo(address address.Address, miner address.Address, sectorSize abi.SectorSize, peer peer.ID, addrs []abi.Multiaddrs) storagemarket.StorageProviderInfo { +func NewStorageProviderInfo(address address.Address, miner address.Address, sectorSize abi.SectorSize, peer peer.ID, addrs []abi.Multiaddrs) legacytypes.StorageProviderInfo { multiaddrs := make([]multiaddr.Multiaddr, 0, len(addrs)) for _, a := range addrs { maddr, err := multiaddr.NewMultiaddrBytes(a) if err != nil { - return storagemarket.StorageProviderInfo{} + return legacytypes.StorageProviderInfo{} } multiaddrs = append(multiaddrs, maddr) } - return storagemarket.StorageProviderInfo{ + return legacytypes.StorageProviderInfo{ Address: address, Worker: miner, SectorSize: uint64(sectorSize), @@ -31,8 +31,8 @@ func NewStorageProviderInfo(address address.Address, miner address.Address, sect } } -func ToSharedBalance(bal api.MarketBalance) storagemarket.Balance { - return storagemarket.Balance{ +func ToSharedBalance(bal api.MarketBalance) legacytypes.Balance { + return legacytypes.Balance{ Locked: bal.Locked, Available: big.Sub(bal.Escrow, bal.Locked), } diff --git a/markets/utils/selectors.go b/markets/utils/selectors.go index e1009d1ff..ec6a0426d 100644 --- a/markets/utils/selectors.go +++ b/markets/utils/selectors.go @@ -7,20 +7,41 @@ import ( "io" // must be imported to init() raw-codec support + dagpb "github.com/ipld/go-codec-dagpb" _ "github.com/ipld/go-ipld-prime/codec/raw" + "github.com/ipld/go-ipld-prime/linking" + "github.com/ipld/go-ipld-prime/node/basicnode" "github.com/ipfs/go-cid" mdagipld "github.com/ipfs/go-ipld-format" "github.com/ipfs/go-unixfsnode" - dagpb "github.com/ipld/go-codec-dagpb" "github.com/ipld/go-ipld-prime" cidlink "github.com/ipld/go-ipld-prime/linking/cid" - basicnode "github.com/ipld/go-ipld-prime/node/basic" "github.com/ipld/go-ipld-prime/traversal" "github.com/ipld/go-ipld-prime/traversal/selector" selectorparse "github.com/ipld/go-ipld-prime/traversal/selector/parse" ) +func CreateLinkSystem(ds mdagipld.DAGService) linking.LinkSystem { + // this is how we implement GETs + linkSystem := cidlink.DefaultLinkSystem() + linkSystem.StorageReadOpener = func(lctx ipld.LinkContext, lnk ipld.Link) (io.Reader, error) { + cl, isCid := lnk.(cidlink.Link) + if !isCid { + return nil, fmt.Errorf("unexpected link type %#v", lnk) + } + + node, err := ds.Get(lctx.Ctx, cl.Cid) + if err != nil { + return nil, err + } + + return bytes.NewBuffer(node.RawData()), nil + } + unixfsnode.AddUnixFSReificationToLinkSystem(&linkSystem) + return linkSystem +} + func TraverseDag( ctx context.Context, ds mdagipld.DAGService, @@ -38,9 +59,8 @@ func TraverseDag( return err } - // not sure what this is for TBH: we also provide ctx in &traversal.Config{} linkContext := ipld.LinkContext{Ctx: ctx} - + linkSystem := CreateLinkSystem(ds) // this is what allows us to understand dagpb nodePrototypeChooser := dagpb.AddSupportToChooser( func(ipld.Link, ipld.LinkContext) (ipld.NodePrototype, error) { @@ -48,23 +68,6 @@ func TraverseDag( }, ) - // this is how we implement GETs - linkSystem := cidlink.DefaultLinkSystem() - linkSystem.StorageReadOpener = func(lctx ipld.LinkContext, lnk ipld.Link) (io.Reader, error) { - cl, isCid := lnk.(cidlink.Link) - if !isCid { - return nil, fmt.Errorf("unexpected link type %#v", lnk) - } - - node, err := ds.Get(lctx.Ctx, cl.Cid) - if err != nil { - return nil, err - } - - return bytes.NewBuffer(node.RawData()), nil - } - unixfsnode.AddUnixFSReificationToLinkSystem(&linkSystem) - // this is how we pull the start node out of the DS startLink := cidlink.Link{Cid: startFrom} startNodePrototype, err := nodePrototypeChooser(startLink, linkContext) diff --git a/node/builder.go b/node/builder.go index 62a81959a..4ae61cde6 100644 --- a/node/builder.go +++ b/node/builder.go @@ -7,11 +7,6 @@ import ( "fmt" "time" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - rmnet "github.com/filecoin-project/boost-gfm/retrievalmarket/network" - gfm_storagemarket "github.com/filecoin-project/boost-gfm/storagemarket" - storageimpl "github.com/filecoin-project/boost-gfm/storagemarket/impl" - "github.com/filecoin-project/boost-gfm/storagemarket/impl/storedask" "github.com/filecoin-project/boost/api" "github.com/filecoin-project/boost/build" "github.com/filecoin-project/boost/cmd/lib" @@ -24,7 +19,6 @@ import ( "github.com/filecoin-project/boost/lib/legacy" "github.com/filecoin-project/boost/lib/mpoolmonitor" "github.com/filecoin-project/boost/markets/idxprov" - "github.com/filecoin-project/boost/markets/retrievaladapter" "github.com/filecoin-project/boost/markets/storageadapter" "github.com/filecoin-project/boost/node/config" "github.com/filecoin-project/boost/node/impl" @@ -43,19 +37,18 @@ import ( "github.com/filecoin-project/boost/storagemarket" "github.com/filecoin-project/boost/storagemarket/dealfilter" "github.com/filecoin-project/boost/storagemarket/sealingpipeline" + "github.com/filecoin-project/boost/storagemarket/storedask" smtypes "github.com/filecoin-project/boost/storagemarket/types" "github.com/filecoin-project/dagstore" "github.com/filecoin-project/go-address" - lotus_gfm_storagemarket "github.com/filecoin-project/go-fil-markets/storagemarket" "github.com/filecoin-project/go-state-types/abi" + "github.com/filecoin-project/go-statemachine/fsm" lotus_api "github.com/filecoin-project/lotus/api" "github.com/filecoin-project/lotus/chain/types" lotus_journal "github.com/filecoin-project/lotus/journal" "github.com/filecoin-project/lotus/journal/alerting" _ "github.com/filecoin-project/lotus/lib/sigs/bls" _ "github.com/filecoin-project/lotus/lib/sigs/secp" - mdagstore "github.com/filecoin-project/lotus/markets/dagstore" - lotus_dealfilter "github.com/filecoin-project/lotus/markets/dealfilter" lotus_config "github.com/filecoin-project/lotus/node/config" lotus_common "github.com/filecoin-project/lotus/node/impl/common" lotus_net "github.com/filecoin-project/lotus/node/impl/net" @@ -138,47 +131,22 @@ const ( StartListeningKey BootstrapKey - // filecoin - SetGenesisKey - - RunHelloKey - RunChainExchangeKey - RunChainGraphsync - RunPeerMgrKey - - HandleIncomingBlocksKey - HandleIncomingMessagesKey - HandleMigrateClientFundsKey - HandlePaymentChannelManagerKey - // miner - GetParamsKey + StartProviderDataTransferKey StartPieceDoctorKey - HandleMigrateProviderFundsKey - HandleDealsKey HandleCreateRetrievalTablesKey - HandleSetShardSelector - HandleSetRetrievalAskGetter HandleRetrievalEventsKey - HandleRetrievalKey HandleRetrievalAskKey HandleRetrievalTransportsKey HandleProtocolProxyKey - RunSectorServiceKey // boost should be started after legacy markets (HandleDealsKey) HandleBoostDealsKey HandleContractDealsKey HandleProposalLogCleanerKey - HandleOnlineBackupMgrKey // daemon ExtractApiKey - HeadMetricsKey - SettlePaymentChannelsKey - RunPeerTaggerKey - SetupFallbackBlockstoresKey - HandleSetLinkSystem SetApiEndpointKey @@ -440,22 +408,11 @@ var BoostNode = Options( Override(new(*db.ProposalLogsDB), modules.NewProposalLogsDB), Override(new(*db.FundsDB), modules.NewFundsDB), Override(new(*db.SectorStateDB), modules.NewSectorStateDB), + Override(new(*storedask.StorageAskDB), storedask.NewStorageAskDB), Override(new(*rtvllog.RetrievalLogDB), modules.NewRetrievalLogDB), ) func ConfigBoost(cfg *config.Boost) Option { - pricingConfig := cfg.Dealmaking.RetrievalPricing - if pricingConfig.Strategy == config.RetrievalPricingExternalMode { - if pricingConfig.External == nil { - return Error(errors.New("retrieval pricing policy has been to set to external but external policy config is nil")) - } - - if pricingConfig.External.Path == "" { - return Error(errors.New("retrieval pricing policy has been to set to external but external script path is empty")) - } - } else if pricingConfig.Strategy != config.RetrievalPricingDefaultMode { - return Error(errors.New("retrieval pricing policy must be either default or external")) - } collatWalletStr := cfg.Wallets.DealCollateral if collatWalletStr == "" && cfg.Wallets.PledgeCollateral != "" { // nolint:staticcheck @@ -473,16 +430,11 @@ func ConfigBoost(cfg *config.Boost) Option { if err != nil { return Error(fmt.Errorf("failed to parse cfg.Wallets.Miner: %s; err: %w", cfg.Wallets.Miner, err)) } - if len(cfg.DAGStore.RootDir) > 0 { - return Error(fmt.Errorf("Detected custom DAG store path %s. The DAG store must be at $BOOST_PATH/dagstore", cfg.DAGStore.RootDir)) - } if cfg.HttpDownload.NChunks < 1 || cfg.HttpDownload.NChunks > 16 { return Error(errors.New("HttpDownload.NChunks should be between 1 and 16")) } - legacyFees := cfg.LotusFees.Legacy() - return Options( ConfigCommon(&cfg.Common), @@ -502,7 +454,7 @@ func ConfigBoost(cfg *config.Boost) Option { StorageMiner: walletMiner, CollatWallet: walletDealCollat, PubMsgWallet: walletPSD, - PubMsgBalMin: abi.TokenAmount(cfg.LotusFees.MaxPublishDealsFee), + PubMsgBalMin: abi.TokenAmount(cfg.Dealpublish.MaxPublishDealsFee), })), Override(new(*storagemanager.StorageManager), storagemanager.New(storagemanager.Config{ @@ -520,8 +472,9 @@ func ConfigBoost(cfg *config.Boost) Option { Override(new(*sectorstatemgr.SectorStateMgr), sectorstatemgr.NewSectorStateMgr(cfg)), Override(new(*indexprovider.Wrapper), indexprovider.NewWrapper(cfg)), + Override(new(storedask.StoredAsk), storedask.NewStoredAsk(cfg)), - Override(new(*legacy.LegacyDealsManager), modules.NewLegacyDealsManager), + Override(new(legacy.LegacyDealManager), modules.NewLegacyDealsManager), Override(new(*storagemarket.ChainDealManager), modules.NewChainDealManager), Override(new(smtypes.CommpCalculator), From(new(lotus_modules.MinerStorageService))), @@ -531,8 +484,8 @@ func ConfigBoost(cfg *config.Boost) Option { Override(new(*mpoolmonitor.MpoolMonitor), modules.NewMpoolMonitor(cfg)), // GraphQL server - Override(new(gql.BlockGetter), modules.NewBlockGetter), - Override(new(*gql.Server), modules.NewGraphqlServer(cfg)), + Override(new(gql.BlockGetter), gql.NewBlockGetter), + Override(new(*gql.Server), gql.NewGraphqlServer(cfg)), // Tracing Override(new(*tracing.Tracing), modules.NewTracing(cfg)), @@ -543,52 +496,24 @@ func ConfigBoost(cfg *config.Boost) Option { })), // Lotus Markets + Override(new(dtypes.ProviderTransport), modules.NewProviderTransport), Override(new(dtypes.ProviderTransferNetwork), modules.NewProviderTransferNetwork), - Override(new(*modules.ProxyAskGetter), modules.NewAskGetter), - Override(new(server.AskGetter), From(new(*modules.ProxyAskGetter))), - Override(new(*modules.LinkSystemProv), modules.NewLinkSystemProvider), - Override(new(server.LinkSystemProvider), From(new(*modules.LinkSystemProv))), - Override(new(*server.GraphsyncUnpaidRetrieval), modules.RetrievalGraphsync(cfg.LotusDealmaking.SimultaneousTransfersForStorage, cfg.LotusDealmaking.SimultaneousTransfersForStoragePerClient, cfg.LotusDealmaking.SimultaneousTransfersForRetrieval)), + Override(StartProviderDataTransferKey, server.NewProviderDataTransfer), + Override(new(server.RetrievalAskGetter), server.NewRetrievalAskGetter), + Override(new(*server.GraphsyncUnpaidRetrieval), modules.RetrievalGraphsync(cfg.Retrievals.Graphsync.SimultaneousTransfersForRetrieval)), Override(new(dtypes.StagingGraphsync), From(new(*server.GraphsyncUnpaidRetrieval))), - Override(new(dtypes.ProviderPieceStore), modules.NewProviderPieceStore), Override(StartPieceDoctorKey, modules.NewPieceDoctor(cfg)), // Lotus Markets (retrieval deps) Override(new(sealer.PieceProvider), sealer.NewPieceProvider), - - Override(new(dtypes.RetrievalPricingFunc), modules.RetrievalPricingFunc(config.DealmakingConfig{ - RetrievalPricing: &lotus_config.RetrievalPricing{ - Strategy: config.RetrievalPricingDefaultMode, - Default: &lotus_config.RetrievalPricingDefault{}, - }, - })), - - // DAG Store - - // TODO: Not sure how to completely get rid of these yet: - // Error: creating node: starting node: missing dependencies for function "reflect".makeFuncStub (/usr/local/go/src/reflect/asm_amd64.s:30): missing types: *dagstore.DAGStore; *dagstore.Wrapper (did you mean stores.DAGStoreWrapper?) - Override(new(*dagstore.DAGStore), func() *dagstore.DAGStore { return nil }), - Override(new(*mdagstore.Wrapper), func() *mdagstore.Wrapper { return nil }), - Override(new(*bdclient.Store), modules.NewPieceDirectoryStore(cfg)), Override(new(*lib.MultiMinerAccessor), modules.NewMultiminerSectorAccessor(cfg)), Override(new(*piecedirectory.PieceDirectory), modules.NewPieceDirectory(cfg)), - Override(DAGStoreKey, modules.NewDAGStoreWrapper), Override(new(dagstore.Interface), From(new(*dagstore.DAGStore))), - Override(new(*modules.ShardSelector), modules.NewShardSelector), - Override(new(dtypes.IndexBackedBlockstore), modules.NewIndexBackedBlockstore(cfg)), - Override(HandleSetShardSelector, modules.SetShardSelectorFunc), - // Lotus Markets (retrieval) - Override(new(mdagstore.SectorAccessor), modules.NewSectorAccessor(cfg)), - Override(new(retrievalmarket.SectorAccessor), From(new(mdagstore.SectorAccessor))), - Override(new(retrievalmarket.RetrievalProviderNode), retrievaladapter.NewRetrievalProviderNode), - Override(new(rmnet.RetrievalMarketNetwork), modules.RetrievalNetwork), - Override(new(retrievalmarket.RetrievalProvider), modules.RetrievalProvider), - Override(HandleSetRetrievalAskGetter, modules.SetAskGetter), - Override(HandleRetrievalEventsKey, modules.HandleRetrievalGraphsyncUpdates(time.Duration(cfg.Dealmaking.RetrievalLogDuration), time.Duration(cfg.Dealmaking.StalledRetrievalTimeout))), - Override(HandleRetrievalKey, modules.HandleRetrieval), + Override(new(server.SectorAccessor), modules.NewSectorAccessor(cfg)), + Override(HandleRetrievalEventsKey, modules.HandleRetrievalGraphsyncUpdates(time.Duration(cfg.Retrievals.Graphsync.RetrievalLogDuration), time.Duration(cfg.Retrievals.Graphsync.StalledRetrievalTimeout))), Override(HandleRetrievalAskKey, modules.HandleQueryAsk), Override(new(*lp2pimpl.TransportsListener), modules.NewTransportsListener(cfg)), Override(new(*protocolproxy.ProtocolProxy), modules.NewProtocolProxy(cfg)), @@ -598,17 +523,10 @@ func ConfigBoost(cfg *config.Boost) Option { Override(new(provider.Interface), modules.IndexProvider(cfg.IndexProvider)), // Lotus Markets (storage) - Override(new(dtypes.ProviderTransport), modules.NewProviderTransport), - Override(new(dtypes.ProviderDataTransfer), modules.NewProviderDataTransfer), - Override(new(*storedask.StoredAsk), modules.NewStorageAsk), - - Override(new(gfm_storagemarket.StorageProviderNode), storageadapter.NewProviderNodeAdapter(&legacyFees, &cfg.LotusDealmaking)), - Override(new(gfm_storagemarket.StorageProvider), modules.NewLegacyStorageProvider(cfg)), - Override(HandleDealsKey, modules.HandleLegacyDeals), + Override(new(fsm.Group), modules.NewLegacyDealsFSM(cfg)), Override(HandleBoostDealsKey, modules.HandleBoostLibp2pDeals(cfg)), Override(HandleContractDealsKey, modules.HandleContractDeals(&cfg.ContractDeals)), Override(HandleProposalLogCleanerKey, modules.HandleProposalLogCleaner(time.Duration(cfg.Dealmaking.DealProposalLogDuration))), - Override(HandleSetLinkSystem, modules.SetLinkSystem), // Boost storage deal filter Override(new(dtypes.StorageDealFilter), modules.BasicDealFilter(nil)), @@ -616,31 +534,17 @@ func ConfigBoost(cfg *config.Boost) Option { Override(new(dtypes.StorageDealFilter), modules.BasicDealFilter(dtypes.StorageDealFilter(dealfilter.CliStorageDealFilter(cfg.Dealmaking.Filter)))), ), - // Lotus markets storage deal filter - Override(new(lotus_dtypes.StorageDealFilter), lotus_modules.BasicDealFilter(cfg.LotusDealmaking, nil)), - If(cfg.LotusDealmaking.Filter != "", - Override(new(lotus_dtypes.StorageDealFilter), lotus_modules.BasicDealFilter(cfg.LotusDealmaking, lotus_dealfilter.CliStorageDealFilter(cfg.LotusDealmaking.Filter))), - ), - Override(new(storageimpl.DealDeciderFunc), modules.DealDeciderFn), - // Boost retrieval deal filter Override(new(dtypes.RetrievalDealFilter), modules.RetrievalDealFilter(nil)), - If(cfg.Dealmaking.RetrievalFilter != "", - Override(new(dtypes.RetrievalDealFilter), modules.RetrievalDealFilter(dtypes.RetrievalDealFilter(dealfilter.CliRetrievalDealFilter(cfg.Dealmaking.RetrievalFilter)))), + If(cfg.Retrievals.Graphsync.RetrievalFilter != "", + Override(new(dtypes.RetrievalDealFilter), modules.RetrievalDealFilter(dtypes.RetrievalDealFilter(dealfilter.CliRetrievalDealFilter(cfg.Retrievals.Graphsync.RetrievalFilter)))), ), - // Lotus markets retrieval deal filter - Override(new(lotus_gfm_storagemarket.StorageProviderNode), modules.LotusGFMStorageProviderNode), - Override(new(lotus_dtypes.RetrievalDealFilter), lotus_modules.RetrievalDealFilter(nil)), - If(cfg.LotusDealmaking.RetrievalFilter != "", - Override(new(lotus_dtypes.RetrievalDealFilter), lotus_modules.RetrievalDealFilter(lotus_dealfilter.CliRetrievalDealFilter(cfg.LotusDealmaking.RetrievalFilter))), - ), - - Override(new(*storageadapter.DealPublisher), storageadapter.NewDealPublisher(&legacyFees, storageadapter.PublishMsgConfig{ - Period: time.Duration(cfg.LotusDealmaking.PublishMsgPeriod), - MaxDealsPerMsg: cfg.LotusDealmaking.MaxDealsPerPublishMsg, + Override(new(*storageadapter.DealPublisher), storageadapter.NewDealPublisher(&cfg.Dealpublish.MaxPublishDealsFee, storageadapter.PublishMsgConfig{ + Period: time.Duration(cfg.Dealpublish.PublishMsgPeriod), + MaxDealsPerMsg: cfg.Dealpublish.MaxDealsPerPublishMsg, StartEpochSealingBuffer: cfg.Dealmaking.StartEpochSealingBuffer, - ManualDealPublish: cfg.Dealmaking.ManualDealPublish, + ManualDealPublish: cfg.Dealpublish.ManualDealPublish, })), Override(new(sealer.Unsealer), From(new(lotus_modules.MinerStorageService))), @@ -652,26 +556,6 @@ func ConfigBoost(cfg *config.Boost) Option { Override(new(sealer.StorageAuth), lotus_modules.StorageAuthWithURL(cfg.SectorIndexApiInfo)), Override(new(*backupmgr.BackupMgr), modules.NewOnlineBackupMgr(cfg)), - // Dynamic Lotus configs - Override(new(lotus_dtypes.ConsiderOnlineStorageDealsConfigFunc), lotus_modules.NewConsiderOnlineStorageDealsConfigFunc), - Override(new(lotus_dtypes.SetConsiderOnlineStorageDealsConfigFunc), lotus_modules.NewSetConsideringOnlineStorageDealsFunc), - Override(new(lotus_dtypes.ConsiderOnlineRetrievalDealsConfigFunc), lotus_modules.NewConsiderOnlineRetrievalDealsConfigFunc), - Override(new(lotus_dtypes.SetConsiderOnlineRetrievalDealsConfigFunc), lotus_modules.NewSetConsiderOnlineRetrievalDealsConfigFunc), - Override(new(lotus_dtypes.StorageDealPieceCidBlocklistConfigFunc), lotus_modules.NewStorageDealPieceCidBlocklistConfigFunc), - Override(new(lotus_dtypes.SetStorageDealPieceCidBlocklistConfigFunc), lotus_modules.NewSetStorageDealPieceCidBlocklistConfigFunc), - Override(new(lotus_dtypes.ConsiderOfflineStorageDealsConfigFunc), lotus_modules.NewConsiderOfflineStorageDealsConfigFunc), - Override(new(lotus_dtypes.SetConsiderOfflineStorageDealsConfigFunc), lotus_modules.NewSetConsideringOfflineStorageDealsFunc), - Override(new(lotus_dtypes.ConsiderOfflineRetrievalDealsConfigFunc), lotus_modules.NewConsiderOfflineRetrievalDealsConfigFunc), - Override(new(lotus_dtypes.SetConsiderOfflineRetrievalDealsConfigFunc), lotus_modules.NewSetConsiderOfflineRetrievalDealsConfigFunc), - Override(new(lotus_dtypes.ConsiderVerifiedStorageDealsConfigFunc), lotus_modules.NewConsiderVerifiedStorageDealsConfigFunc), - Override(new(lotus_dtypes.SetConsiderVerifiedStorageDealsConfigFunc), lotus_modules.NewSetConsideringVerifiedStorageDealsFunc), - Override(new(lotus_dtypes.ConsiderUnverifiedStorageDealsConfigFunc), lotus_modules.NewConsiderUnverifiedStorageDealsConfigFunc), - Override(new(lotus_dtypes.SetConsiderUnverifiedStorageDealsConfigFunc), lotus_modules.NewSetConsideringUnverifiedStorageDealsFunc), - Override(new(lotus_dtypes.SetExpectedSealDurationFunc), lotus_modules.NewSetExpectedSealDurationFunc), - Override(new(lotus_dtypes.GetExpectedSealDurationFunc), lotus_modules.NewGetExpectedSealDurationFunc), - Override(new(lotus_dtypes.SetMaxDealStartDelayFunc), lotus_modules.NewSetMaxDealStartDelayFunc), - Override(new(lotus_dtypes.GetMaxDealStartDelayFunc), lotus_modules.NewGetMaxDealStartDelayFunc), - // Dynamic Boost configs Override(new(dtypes.ConsiderOnlineStorageDealsConfigFunc), modules.NewConsiderOnlineStorageDealsConfigFunc), Override(new(dtypes.SetConsiderOnlineStorageDealsConfigFunc), modules.NewSetConsideringOnlineStorageDealsFunc), diff --git a/node/config/def.go b/node/config/def.go index 9f2855689..817e7411f 100644 --- a/node/config/def.go +++ b/node/config/def.go @@ -11,14 +11,6 @@ import ( "github.com/ipfs/go-cid" ) -const ( - // RetrievalPricingDefault configures the node to use the default retrieval pricing policy. - RetrievalPricingDefaultMode = "default" - // RetrievalPricingExternal configures the node to use the external retrieval pricing script - // configured by the user. - RetrievalPricingExternalMode = "external" -) - // MaxTraversalLinks configures the maximum number of links to traverse in a DAG while calculating // CommP and traversing a DAG with graphsync; invokes a budget on DAG depth and density. var MaxTraversalLinks uint64 = 32 * (1 << 20) @@ -55,7 +47,6 @@ func defCommon() Common { } -var DefaultDefaultMaxFee = types.MustParseFIL("0.07") var DefaultSimultaneousTransfers = uint64(20) func DefaultBoost() *Boost { @@ -105,98 +96,27 @@ func DefaultBoost() *Boost { }, Dealmaking: DealmakingConfig{ - ConsiderOnlineStorageDeals: true, - ConsiderOfflineStorageDeals: true, - ConsiderOnlineRetrievalDeals: true, - ConsiderOfflineRetrievalDeals: true, - ConsiderVerifiedStorageDeals: true, - ConsiderUnverifiedStorageDeals: true, - PieceCidBlocklist: []cid.Cid{}, - // TODO: It'd be nice to set this based on sector size + ConsiderOnlineStorageDeals: true, + ConsiderOfflineStorageDeals: true, + ConsiderOnlineRetrievalDeals: true, + ConsiderOfflineRetrievalDeals: true, + ConsiderVerifiedStorageDeals: true, + ConsiderUnverifiedStorageDeals: true, + PieceCidBlocklist: []cid.Cid{}, MaxDealStartDelay: Duration(time.Hour * 24 * 14), ExpectedSealDuration: Duration(time.Hour * 24), MaxProviderCollateralMultiplier: 2, - - StartEpochSealingBuffer: 480, // 480 epochs buffer == 4 hours from adding deal to sector to sector being sealed - - DealProposalLogDuration: Duration(time.Hour * 24), - RetrievalLogDuration: Duration(time.Hour * 24), - StalledRetrievalTimeout: Duration(time.Second * 30), - - RetrievalPricing: &lotus_config.RetrievalPricing{ - Strategy: RetrievalPricingDefaultMode, - Default: &lotus_config.RetrievalPricingDefault{ - VerifiedDealsFreeTransfer: true, - }, - External: &lotus_config.RetrievalPricingExternal{ - Path: "", - }, - }, - - // This should no longer be needed once LID is live - BlockstoreCacheMaxShards: 20, // Match default simultaneous retrievals - BlockstoreCacheExpiry: Duration(30 * time.Second), - - IsUnsealedCacheExpiry: Duration(5 * time.Minute), - - MaxTransferDuration: Duration(24 * 3600 * time.Second), - - RemoteCommp: false, - MaxConcurrentLocalCommp: 1, - - HttpTransferMaxConcurrentDownloads: 20, - HttpTransferStallTimeout: Duration(5 * time.Minute), - HttpTransferStallCheckPeriod: Duration(30 * time.Second), - DealLogDurationDays: 30, - SealingPipelineCacheTimeout: Duration(30 * time.Second), - FundsTaggingEnabled: true, - EnableLegacyStorageDeals: false, - ManualDealPublish: false, - BitswapPublicAddresses: []string{}, + StartEpochSealingBuffer: 480, // 480 epochs buffer == 4 hours from adding deal to sector to sector being sealed + DealProposalLogDuration: Duration(time.Hour * 24), + IsUnsealedCacheExpiry: Duration(5 * time.Minute), + MaxTransferDuration: Duration(24 * 3600 * time.Second), + RemoteCommp: false, + MaxConcurrentLocalCommp: 1, + DealLogDurationDays: 30, + SealingPipelineCacheTimeout: Duration(30 * time.Second), + FundsTaggingEnabled: true, }, - LotusDealmaking: lotus_config.DealmakingConfig{ - ConsiderOnlineStorageDeals: true, - ConsiderOfflineStorageDeals: true, - ConsiderOnlineRetrievalDeals: true, - ConsiderOfflineRetrievalDeals: true, - ConsiderVerifiedStorageDeals: true, - ConsiderUnverifiedStorageDeals: true, - PieceCidBlocklist: []cid.Cid{}, - // TODO: It'd be nice to set this based on sector size - MaxDealStartDelay: lotus_config.Duration(time.Hour * 24 * 14), - ExpectedSealDuration: lotus_config.Duration(time.Hour * 24), - PublishMsgPeriod: lotus_config.Duration(time.Hour), - MaxDealsPerPublishMsg: 8, - MaxProviderCollateralMultiplier: 2, - - SimultaneousTransfersForStorage: DefaultSimultaneousTransfers, - SimultaneousTransfersForStoragePerClient: 0, - SimultaneousTransfersForRetrieval: DefaultSimultaneousTransfers, - - StartEpochSealingBuffer: 480, // 480 epochs buffer == 4 hours from adding deal to sector to sector being sealed - - RetrievalPricing: &lotus_config.RetrievalPricing{ - Strategy: RetrievalPricingDefaultMode, - Default: &lotus_config.RetrievalPricingDefault{ - VerifiedDealsFreeTransfer: true, - }, - External: &lotus_config.RetrievalPricingExternal{ - Path: "", - }, - }, - }, - - LotusFees: FeeConfig{ - MaxPublishDealsFee: types.MustParseFIL("0.05"), - MaxMarketBalanceAddFee: types.MustParseFIL("0.007"), - }, - - DAGStore: lotus_config.DAGStoreConfig{ - MaxConcurrentIndex: 5, - MaxConcurrencyStorageCalls: 100, - GCInterval: lotus_config.Duration(1 * time.Minute), - }, IndexProvider: IndexProviderConfig{ Enable: true, EntriesCacheCapacity: 1024, @@ -223,8 +143,31 @@ func DefaultBoost() *Boost { DataTransferPublisher: false, }, HttpDownload: HttpDownloadConfig{ - NChunks: 5, - AllowPrivateIPs: false, + HttpTransferMaxConcurrentDownloads: 20, + HttpTransferStallTimeout: Duration(5 * time.Minute), + HttpTransferStallCheckPeriod: Duration(30 * time.Second), + NChunks: 5, + AllowPrivateIPs: false, + }, + Retrievals: RetrievalConfig{ + Graphsync: GraphsyncRetrievalConfig{ + SimultaneousTransfersForRetrieval: DefaultSimultaneousTransfers, + RetrievalLogDuration: Duration(time.Hour * 24), + StalledRetrievalTimeout: Duration(time.Second * 30), + GraphsyncStorageAccessApiInfo: []string{}, + }, + Bitswap: BitswapRetrievalConfig{ + BitswapPublicAddresses: []string{}, + }, + HTTP: HTTPRetrievalConfig{ + HTTPRetrievalMultiaddr: "", + }, + }, + Dealpublish: DealPublishConfig{ + ManualDealPublish: false, + PublishMsgPeriod: Duration(time.Hour), + MaxDealsPerPublishMsg: 8, + MaxPublishDealsFee: types.MustParseFIL("0.05"), }, } return cfg diff --git a/node/config/doc_gen.go b/node/config/doc_gen.go index 96b0e5dbb..64d734e2c 100644 --- a/node/config/doc_gen.go +++ b/node/config/doc_gen.go @@ -18,6 +18,44 @@ var Doc = map[string][]DocField{ your node if metadata log is disabled`, }, }, + "BitswapRetrievalConfig": []DocField{ + { + Name: "BitswapPeerID", + Type: "string", + + Comment: `The libp2p peer id used by booster-bitswap. +Run 'booster-bitswap init' to get the peer id. +When BitswapPeerID is not empty boostd will: +- listen on bitswap protocols on boostd's own peer id and proxy +requests to booster-bitswap +- advertise boostd's peer id in bitswap records to the content indexer +(bitswap clients connect to boostd, which proxies the requests to +booster-bitswap) +- list bitswap as an available transport on the retrieval transport protocol`, + }, + { + Name: "BitswapPublicAddresses", + Type: "[]string", + + Comment: `Public multiaddresses for booster-bitswap. +If empty +- booster-bitswap is assumed to be running privately +- boostd acts as a proxy: it listens on bitswap protocols on boostd's own +peer id and forwards them to booster-bitswap +If public addresses are set +- boostd announces the booster-bitswap peer id to the indexer as an +extended provider +- clients make connections directly to the booster-bitswap process +(boostd does not act as a proxy)`, + }, + { + Name: "BitswapPrivKeyFile", + Type: "string", + + Comment: `If operating in public mode, in order to announce booster-bitswap as an extended provider, this value must point to a +a file containing the booster-bitswap peer id's private key. Can be left blank when operating with protocol proxy.`, + }, + }, "Boost": []DocField{ { Name: "ConfigVersion", @@ -49,6 +87,12 @@ your node if metadata log is disabled`, Comment: ``, }, + { + Name: "Dealpublish", + Type: "DealPublishConfig", + + Comment: ``, + }, { Name: "Wallets", Type: "WalletsConfig", @@ -92,20 +136,8 @@ your node if metadata log is disabled`, Comment: ``, }, { - Name: "LotusDealmaking", - Type: "lotus_config.DealmakingConfig", - - Comment: `Lotus configs`, - }, - { - Name: "LotusFees", - Type: "FeeConfig", - - Comment: ``, - }, - { - Name: "DAGStore", - Type: "lotus_config.DAGStoreConfig", + Name: "Retrievals", + Type: "RetrievalConfig", Comment: ``, }, @@ -162,6 +194,38 @@ your node if metadata log is disabled`, Comment: `From address for eth_ state call`, }, }, + "DealPublishConfig": []DocField{ + { + Name: "ManualDealPublish", + Type: "bool", + + Comment: `When set to true, the user is responsible for publishing deals manually. +The values of MaxDealsPerPublishMsg and PublishMsgPeriod will be +ignored, and deals will remain in the pending state until manually published.`, + }, + { + Name: "PublishMsgPeriod", + Type: "Duration", + + Comment: `When a deal is ready to publish, the amount of time to wait for more +deals to be ready to publish before publishing them all as a batch`, + }, + { + Name: "MaxDealsPerPublishMsg", + Type: "uint64", + + Comment: `The maximum number of deals to include in a single PublishStorageDeals +message`, + }, + { + Name: "MaxPublishDealsFee", + Type: "types.FIL", + + Comment: `The maximum collateral that the provider will put up against a deal, +as a multiplier of the minimum collateral bound +The maximum fee to pay when sending the PublishStorageDeals message`, + }, + }, "DealmakingConfig": []DocField{ { Name: "ConsiderOnlineStorageDeals", @@ -267,20 +331,6 @@ Set this value to 0 to indicate there is no limit per host.`, Comment: `The amount of time to keep deal proposal logs for before cleaning them up.`, }, - { - Name: "RetrievalLogDuration", - Type: "Duration", - - Comment: `The amount of time to keep retrieval deal logs for before cleaning them up. -Note RetrievalLogDuration should exceed the StalledRetrievalTimeout as the -logs db is leveraged for pruning stalled retrievals.`, - }, - { - Name: "StalledRetrievalTimeout", - Type: "Duration", - - Comment: `The amount of time stalled retrieval deals will remain open before being canceled.`, - }, { Name: "Filter", Type: "string", @@ -288,32 +338,6 @@ logs db is leveraged for pruning stalled retrievals.`, Comment: `A command used for fine-grained evaluation of storage deals see https://boost.filecoin.io/configuration/deal-filters for more details`, }, - { - Name: "RetrievalFilter", - Type: "string", - - Comment: `A command used for fine-grained evaluation of retrieval deals -see https://boost.filecoin.io/configuration/deal-filters for more details`, - }, - { - Name: "RetrievalPricing", - Type: "*lotus_config.RetrievalPricing", - - Comment: ``, - }, - { - Name: "BlockstoreCacheMaxShards", - Type: "int", - - Comment: `The maximum number of shards cached by the Dagstore for retrieval -Lower this limit if boostd memory is too high during retrievals`, - }, - { - Name: "BlockstoreCacheExpiry", - Type: "Duration", - - Comment: `How long a blockstore shard should be cached before expiring without use`, - }, { Name: "IsUnsealedCacheExpiry", Type: "Duration", @@ -339,70 +363,6 @@ Please note that this only works for v1.2.0 deals and not legacy deals`, Comment: `The maximum number of commp processes to run in parallel on the local boost process`, - }, - { - Name: "HTTPRetrievalMultiaddr", - Type: "string", - - Comment: `The public multi-address for retrieving deals with booster-http. -Note: Must be in multiaddr format, eg /dns/foo.com/tcp/443/https`, - }, - { - Name: "HttpTransferMaxConcurrentDownloads", - Type: "uint64", - - Comment: `The maximum number of concurrent storage deal HTTP downloads. -Note that this is a soft maximum; if some downloads stall, -more downloads are allowed to start.`, - }, - { - Name: "HttpTransferStallCheckPeriod", - Type: "Duration", - - Comment: `The period between checking if downloads have stalled.`, - }, - { - Name: "HttpTransferStallTimeout", - Type: "Duration", - - Comment: `The time that can elapse before a download is considered stalled (and -another concurrent download is allowed to start).`, - }, - { - Name: "BitswapPeerID", - Type: "string", - - Comment: `The libp2p peer id used by booster-bitswap. -Run 'booster-bitswap init' to get the peer id. -When BitswapPeerID is not empty boostd will: -- listen on bitswap protocols on boostd's own peer id and proxy -requests to booster-bitswap -- advertise boostd's peer id in bitswap records to the content indexer -(bitswap clients connect to boostd, which proxies the requests to -booster-bitswap) -- list bitswap as an available transport on the retrieval transport protocol`, - }, - { - Name: "BitswapPublicAddresses", - Type: "[]string", - - Comment: `Public multiaddresses for booster-bitswap. -If empty -- booster-bitswap is assumed to be running privately -- boostd acts as a proxy: it listens on bitswap protocols on boostd's own -peer id and forwards them to booster-bitswap -If public addresses are set -- boostd announces the booster-bitswap peer id to the indexer as an -extended provider -- clients make connections directly to the booster-bitswap process -(boostd does not act as a proxy)`, - }, - { - Name: "BitswapPrivKeyFile", - Type: "string", - - Comment: `If operating in public mode, in order to announce booster-bitswap as an extended provider, this value must point to a -a file containing the booster-bitswap peer id's private key. Can be left blank when operating with protocol proxy.`, }, { Name: "DealLogDurationDays", @@ -427,20 +387,41 @@ Any value less than 0 will result in use of default`, accepted boost will tag funds for that deal so that they cannot be used for any other deal.`, }, + }, + "GraphqlConfig": []DocField{ { - Name: "EnableLegacyStorageDeals", - Type: "bool", + Name: "ListenAddress", + Type: "string", - Comment: `Whether to enable legacy deals on the Boost node or not. We recommend keeping -them disabled. These will be completely deprecated soon.`, + Comment: `The ip address the GraphQL server will bind to. Default: 127.0.0.1`, }, { - Name: "ManualDealPublish", - Type: "bool", + Name: "Port", + Type: "uint64", - Comment: `When set to true, the user is responsible for publishing deals manually. -The values of MaxDealsPerPublishMsg and PublishMsgPeriod will be -ignored, and deals will remain in the pending state until manually published.`, + Comment: `The port that the graphql server listens on`, + }, + }, + "GraphsyncRetrievalConfig": []DocField{ + { + Name: "SimultaneousTransfersForRetrieval", + Type: "uint64", + + Comment: `The maximum number of parallel online data transfers for retrieval deals`, + }, + { + Name: "RetrievalLogDuration", + Type: "Duration", + + Comment: `The amount of time to keep retrieval deal logs for before cleaning them up. +Note RetrievalLogDuration should exceed the StalledRetrievalTimeout as the +logs db is leveraged for pruning stalled retrievals.`, + }, + { + Name: "StalledRetrievalTimeout", + Type: "Duration", + + Comment: `The amount of time stalled retrieval deals will remain open before being canceled.`, }, { Name: "GraphsyncStorageAccessApiInfo", @@ -451,36 +432,45 @@ sector data from when serving graphsync retrievals. If this parameter is not set, boost will serve data from the endpoint configured in SectorIndexApiInfo.`, }, - }, - "FeeConfig": []DocField{ { - Name: "MaxPublishDealsFee", - Type: "types.FIL", + Name: "RetrievalFilter", + Type: "string", - Comment: `The maximum fee to pay when sending the PublishStorageDeals message`, + Comment: `A command used for fine-grained evaluation of retrieval deals +see https://boost.filecoin.io/configuration/deal-filters for more details`, }, + }, + "HTTPRetrievalConfig": []DocField{ { - Name: "MaxMarketBalanceAddFee", - Type: "types.FIL", + Name: "HTTPRetrievalMultiaddr", + Type: "string", - Comment: `The maximum fee to pay when sending the AddBalance message (used by legacy markets)`, + Comment: `The public multi-address for retrieving deals with booster-http. +Note: Must be in multiaddr format, eg /dns/foo.com/tcp/443/https`, }, }, - "GraphqlConfig": []DocField{ + "HttpDownloadConfig": []DocField{ { - Name: "ListenAddress", - Type: "string", + Name: "HttpTransferMaxConcurrentDownloads", + Type: "uint64", - Comment: `The ip address the GraphQL server will bind to. Default: 127.0.0.1`, + Comment: `The maximum number of concurrent storage deal HTTP downloads. +Note that this is a soft maximum; if some downloads stall, +more downloads are allowed to start.`, }, { - Name: "Port", - Type: "uint64", + Name: "HttpTransferStallCheckPeriod", + Type: "Duration", - Comment: `The port that the graphql server listens on`, + Comment: `The period between checking if downloads have stalled.`, + }, + { + Name: "HttpTransferStallTimeout", + Type: "Duration", + + Comment: `The time that can elapse before a download is considered stalled (and +another concurrent download is allowed to start).`, }, - }, - "HttpDownloadConfig": []DocField{ { Name: "NChunks", Type: "int", @@ -701,112 +691,33 @@ Set this value to "" if the local index directory data service is embedded.`, Comment: `The yugabyte cassandra hosts eg ["127.0.0.1"]`, }, }, - "LotusDealmakingConfig": []DocField{ - { - Name: "PieceCidBlocklist", - Type: "[]cid.Cid", - - Comment: `A list of Data CIDs to reject when making deals`, - }, - { - Name: "ExpectedSealDuration", - Type: "Duration", - - Comment: `Maximum expected amount of time getting the deal into a sealed sector will take -This includes the time the deal will need to get transferred and published -before being assigned to a sector`, - }, - { - Name: "MaxDealStartDelay", - Type: "Duration", - - Comment: `Maximum amount of time proposed deal StartEpoch can be in future`, - }, - { - Name: "PublishMsgPeriod", - Type: "Duration", - - Comment: `When a deal is ready to publish, the amount of time to wait for more -deals to be ready to publish before publishing them all as a batch`, - }, - { - Name: "MaxDealsPerPublishMsg", - Type: "uint64", - - Comment: `The maximum number of deals to include in a single PublishStorageDeals -message`, - }, - { - Name: "MaxProviderCollateralMultiplier", - Type: "uint64", - - Comment: `The maximum collateral that the provider will put up against a deal, -as a multiplier of the minimum collateral bound`, - }, + "MonitoringConfig": []DocField{ { - Name: "MaxStagingDealsBytes", + Name: "MpoolAlertEpochs", Type: "int64", - Comment: `The maximum allowed disk usage size in bytes of staging deals not yet -passed to the sealing node by the markets service. 0 is unlimited.`, - }, - { - Name: "SimultaneousTransfersForStorage", - Type: "uint64", - - Comment: `The maximum number of parallel online data transfers for storage deals`, - }, - { - Name: "SimultaneousTransfersForStoragePerClient", - Type: "uint64", - - Comment: `The maximum number of simultaneous data transfers from any single client -for storage deals. -Unset by default (0), and values higher than SimultaneousTransfersForStorage -will have no effect; i.e. the total number of simultaneous data transfers -across all storage clients is bound by SimultaneousTransfersForStorage -regardless of this number.`, - }, - { - Name: "SimultaneousTransfersForRetrieval", - Type: "uint64", - - Comment: `The maximum number of parallel online data transfers for retrieval deals`, - }, - { - Name: "StartEpochSealingBuffer", - Type: "uint64", - - Comment: `Minimum start epoch buffer to give time for sealing of sector with deal.`, - }, - { - Name: "Filter", - Type: "string", - - Comment: `A command used for fine-grained evaluation of storage deals -see https://boost.filecoin.io/configuration/deal-filters for more details`, + Comment: `The number of epochs after which alert is generated for a local pending +message in lotus mpool`, }, + }, + "RetrievalConfig": []DocField{ { - Name: "RetrievalFilter", - Type: "string", + Name: "Graphsync", + Type: "GraphsyncRetrievalConfig", - Comment: `A command used for fine-grained evaluation of retrieval deals -see https://boost.filecoin.io/configuration/deal-filters for more details`, + Comment: ``, }, { - Name: "RetrievalPricing", - Type: "*lotus_config.RetrievalPricing", + Name: "Bitswap", + Type: "BitswapRetrievalConfig", Comment: ``, }, - }, - "MonitoringConfig": []DocField{ { - Name: "MpoolAlertEpochs", - Type: "int64", + Name: "HTTP", + Type: "HTTPRetrievalConfig", - Comment: `The number of epochs after which alert is generated for a local pending -message in lotus mpool`, + Comment: ``, }, }, "StorageConfig": []DocField{ diff --git a/node/config/migrate.go b/node/config/migrate.go index 67d26b164..849cef3b5 100644 --- a/node/config/migrate.go +++ b/node/config/migrate.go @@ -15,7 +15,7 @@ var log = logging.Logger("cfg") // CurrentVersion is the config version expected by Boost. // We need to migrate the config file to this version. -const CurrentVersion = 5 +const CurrentVersion = 6 type migrateUpFn = func(cfgPath string) (string, error) @@ -25,6 +25,7 @@ var migrations = []migrateUpFn{ v2Tov3, // index 2 => version 3 v3Tov4, // index 3 => version 4 v4Tov5, // index 4 => version 5 + v5Tov6, // index 5 => version 6 } // This struct is used to get the config file version diff --git a/node/config/types.go b/node/config/types.go index 279de42c0..d6717ce39 100644 --- a/node/config/types.go +++ b/node/config/types.go @@ -41,6 +41,7 @@ type Boost struct { SectorIndexApiInfo string Dealmaking DealmakingConfig + Dealpublish DealPublishConfig Wallets WalletsConfig Graphql GraphqlConfig Monitoring MonitoringConfig @@ -48,20 +49,8 @@ type Boost struct { LocalIndexDirectory LocalIndexDirectoryConfig ContractDeals ContractDealsConfig HttpDownload HttpDownloadConfig - - // Lotus configs - LotusDealmaking lotus_config.DealmakingConfig - LotusFees FeeConfig - DAGStore lotus_config.DAGStoreConfig - IndexProvider IndexProviderConfig -} - -func (b *Boost) GetDealmakingConfig() lotus_config.DealmakingConfig { - return b.LotusDealmaking -} - -func (b *Boost) SetDealmakingConfig(other lotus_config.DealmakingConfig) { - b.LotusDealmaking = other + Retrievals RetrievalConfig + IndexProvider IndexProviderConfig } type WalletsConfig struct { @@ -89,51 +78,6 @@ type TracingConfig struct { Endpoint string } -type LotusDealmakingConfig struct { - // A list of Data CIDs to reject when making deals - PieceCidBlocklist []cid.Cid - // Maximum expected amount of time getting the deal into a sealed sector will take - // This includes the time the deal will need to get transferred and published - // before being assigned to a sector - ExpectedSealDuration Duration - // Maximum amount of time proposed deal StartEpoch can be in future - MaxDealStartDelay Duration - // When a deal is ready to publish, the amount of time to wait for more - // deals to be ready to publish before publishing them all as a batch - PublishMsgPeriod Duration - // The maximum number of deals to include in a single PublishStorageDeals - // message - MaxDealsPerPublishMsg uint64 - // The maximum collateral that the provider will put up against a deal, - // as a multiplier of the minimum collateral bound - MaxProviderCollateralMultiplier uint64 - // The maximum allowed disk usage size in bytes of staging deals not yet - // passed to the sealing node by the markets service. 0 is unlimited. - MaxStagingDealsBytes int64 - // The maximum number of parallel online data transfers for storage deals - SimultaneousTransfersForStorage uint64 - // The maximum number of simultaneous data transfers from any single client - // for storage deals. - // Unset by default (0), and values higher than SimultaneousTransfersForStorage - // will have no effect; i.e. the total number of simultaneous data transfers - // across all storage clients is bound by SimultaneousTransfersForStorage - // regardless of this number. - SimultaneousTransfersForStoragePerClient uint64 - // The maximum number of parallel online data transfers for retrieval deals - SimultaneousTransfersForRetrieval uint64 - // Minimum start epoch buffer to give time for sealing of sector with deal. - StartEpochSealingBuffer uint64 - - // A command used for fine-grained evaluation of storage deals - // see https://boost.filecoin.io/configuration/deal-filters for more details - Filter string - // A command used for fine-grained evaluation of retrieval deals - // see https://boost.filecoin.io/configuration/deal-filters for more details - RetrievalFilter string - - RetrievalPricing *lotus_config.RetrievalPricing -} - type DealmakingConfig struct { // When enabled, the miner can accept online deals ConsiderOnlineStorageDeals bool @@ -183,27 +127,10 @@ type DealmakingConfig struct { StartEpochSealingBuffer uint64 // The amount of time to keep deal proposal logs for before cleaning them up. DealProposalLogDuration Duration - // The amount of time to keep retrieval deal logs for before cleaning them up. - // Note RetrievalLogDuration should exceed the StalledRetrievalTimeout as the - // logs db is leveraged for pruning stalled retrievals. - RetrievalLogDuration Duration - // The amount of time stalled retrieval deals will remain open before being canceled. - StalledRetrievalTimeout Duration // A command used for fine-grained evaluation of storage deals // see https://boost.filecoin.io/configuration/deal-filters for more details Filter string - // A command used for fine-grained evaluation of retrieval deals - // see https://boost.filecoin.io/configuration/deal-filters for more details - RetrievalFilter string - - RetrievalPricing *lotus_config.RetrievalPricing - - // The maximum number of shards cached by the Dagstore for retrieval - // Lower this limit if boostd memory is too high during retrievals - BlockstoreCacheMaxShards int - // How long a blockstore shard should be cached before expiring without use - BlockstoreCacheExpiry Duration // How long to cache calls to check whether a sector is unsealed IsUnsealedCacheExpiry Duration @@ -218,47 +145,6 @@ type DealmakingConfig struct { // boost process MaxConcurrentLocalCommp uint64 - // The public multi-address for retrieving deals with booster-http. - // Note: Must be in multiaddr format, eg /dns/foo.com/tcp/443/https - HTTPRetrievalMultiaddr string - - // The maximum number of concurrent storage deal HTTP downloads. - // Note that this is a soft maximum; if some downloads stall, - // more downloads are allowed to start. - HttpTransferMaxConcurrentDownloads uint64 - // The period between checking if downloads have stalled. - HttpTransferStallCheckPeriod Duration - // The time that can elapse before a download is considered stalled (and - // another concurrent download is allowed to start). - HttpTransferStallTimeout Duration - - // The libp2p peer id used by booster-bitswap. - // Run 'booster-bitswap init' to get the peer id. - // When BitswapPeerID is not empty boostd will: - // - listen on bitswap protocols on boostd's own peer id and proxy - // requests to booster-bitswap - // - advertise boostd's peer id in bitswap records to the content indexer - // (bitswap clients connect to boostd, which proxies the requests to - // booster-bitswap) - // - list bitswap as an available transport on the retrieval transport protocol - BitswapPeerID string - - // Public multiaddresses for booster-bitswap. - // If empty - // - booster-bitswap is assumed to be running privately - // - boostd acts as a proxy: it listens on bitswap protocols on boostd's own - // peer id and forwards them to booster-bitswap - // If public addresses are set - // - boostd announces the booster-bitswap peer id to the indexer as an - // extended provider - // - clients make connections directly to the booster-bitswap process - // (boostd does not act as a proxy) - BitswapPublicAddresses []string - - // If operating in public mode, in order to announce booster-bitswap as an extended provider, this value must point to a - // a file containing the booster-bitswap peer id's private key. Can be left blank when operating with protocol proxy. - BitswapPrivKeyFile string - // The deal logs older than DealLogDurationDays are deleted from the logsDB // to keep the size of logsDB in check. Set the value as "0" to disable log cleanup DealLogDurationDays int @@ -272,21 +158,6 @@ type DealmakingConfig struct { // accepted boost will tag funds for that deal so that they cannot be used // for any other deal. FundsTaggingEnabled bool - - // Whether to enable legacy deals on the Boost node or not. We recommend keeping - // them disabled. These will be completely deprecated soon. - EnableLegacyStorageDeals bool - - // When set to true, the user is responsible for publishing deals manually. - // The values of MaxDealsPerPublishMsg and PublishMsgPeriod will be - // ignored, and deals will remain in the pending state until manually published. - ManualDealPublish bool - - // The connect strings for the RPC APIs of each miner that boost can read - // sector data from when serving graphsync retrievals. - // If this parameter is not set, boost will serve data from the endpoint - // configured in SectorIndexApiInfo. - GraphsyncStorageAccessApiInfo []string } type ContractDealsConfig struct { @@ -369,20 +240,6 @@ type IndexProviderHttpPublisherConfig struct { WithLibp2p bool } -type FeeConfig struct { - // The maximum fee to pay when sending the PublishStorageDeals message - MaxPublishDealsFee types.FIL - // The maximum fee to pay when sending the AddBalance message (used by legacy markets) - MaxMarketBalanceAddFee types.FIL -} - -func (c *FeeConfig) Legacy() lotus_config.MinerFeeConfig { - return lotus_config.MinerFeeConfig{ - MaxPublishDealsFee: c.MaxPublishDealsFee, - MaxMarketBalanceAddFee: c.MaxMarketBalanceAddFee, - } -} - type StorageConfig struct { // The maximum number of concurrent fetch operations to the storage subsystem ParallelFetchLimit int @@ -439,6 +296,15 @@ type LocalIndexDirectoryLeveldbConfig struct { } type HttpDownloadConfig struct { + // The maximum number of concurrent storage deal HTTP downloads. + // Note that this is a soft maximum; if some downloads stall, + // more downloads are allowed to start. + HttpTransferMaxConcurrentDownloads uint64 + // The period between checking if downloads have stalled. + HttpTransferStallCheckPeriod Duration + // The time that can elapse before a download is considered stalled (and + // another concurrent download is allowed to start). + HttpTransferStallTimeout Duration // NChunks is a number of chunks to split HTTP downloads into. Each chunk is downloaded in the goroutine of its own // which improves the overall download speed. NChunks is always equal to 1 for libp2p transport because libp2p server // doesn't support range requests yet. NChunks must be greater than 0 and less than 16, with the default of 5. @@ -447,3 +313,81 @@ type HttpDownloadConfig struct { // The default is false. AllowPrivateIPs bool } + +type RetrievalConfig struct { + Graphsync GraphsyncRetrievalConfig + Bitswap BitswapRetrievalConfig + HTTP HTTPRetrievalConfig +} + +type BitswapRetrievalConfig struct { + // The libp2p peer id used by booster-bitswap. + // Run 'booster-bitswap init' to get the peer id. + // When BitswapPeerID is not empty boostd will: + // - listen on bitswap protocols on boostd's own peer id and proxy + // requests to booster-bitswap + // - advertise boostd's peer id in bitswap records to the content indexer + // (bitswap clients connect to boostd, which proxies the requests to + // booster-bitswap) + // - list bitswap as an available transport on the retrieval transport protocol + BitswapPeerID string + + // Public multiaddresses for booster-bitswap. + // If empty + // - booster-bitswap is assumed to be running privately + // - boostd acts as a proxy: it listens on bitswap protocols on boostd's own + // peer id and forwards them to booster-bitswap + // If public addresses are set + // - boostd announces the booster-bitswap peer id to the indexer as an + // extended provider + // - clients make connections directly to the booster-bitswap process + // (boostd does not act as a proxy) + BitswapPublicAddresses []string + + // If operating in public mode, in order to announce booster-bitswap as an extended provider, this value must point to a + // a file containing the booster-bitswap peer id's private key. Can be left blank when operating with protocol proxy. + BitswapPrivKeyFile string +} + +type HTTPRetrievalConfig struct { + // The public multi-address for retrieving deals with booster-http. + // Note: Must be in multiaddr format, eg /dns/foo.com/tcp/443/https + HTTPRetrievalMultiaddr string +} + +type GraphsyncRetrievalConfig struct { + // The maximum number of parallel online data transfers for retrieval deals + SimultaneousTransfersForRetrieval uint64 + // The amount of time to keep retrieval deal logs for before cleaning them up. + // Note RetrievalLogDuration should exceed the StalledRetrievalTimeout as the + // logs db is leveraged for pruning stalled retrievals. + RetrievalLogDuration Duration + // The amount of time stalled retrieval deals will remain open before being canceled. + StalledRetrievalTimeout Duration + // The connect strings for the RPC APIs of each miner that boost can read + // sector data from when serving graphsync retrievals. + // If this parameter is not set, boost will serve data from the endpoint + // configured in SectorIndexApiInfo. + GraphsyncStorageAccessApiInfo []string + // A command used for fine-grained evaluation of retrieval deals + // see https://boost.filecoin.io/configuration/deal-filters for more details + RetrievalFilter string +} + +type DealPublishConfig struct { + // When set to true, the user is responsible for publishing deals manually. + // The values of MaxDealsPerPublishMsg and PublishMsgPeriod will be + // ignored, and deals will remain in the pending state until manually published. + ManualDealPublish bool + + // When a deal is ready to publish, the amount of time to wait for more + // deals to be ready to publish before publishing them all as a batch + PublishMsgPeriod Duration + // The maximum number of deals to include in a single PublishStorageDeals + // message + MaxDealsPerPublishMsg uint64 + // The maximum collateral that the provider will put up against a deal, + // as a multiplier of the minimum collateral bound + // The maximum fee to pay when sending the PublishStorageDeals message + MaxPublishDealsFee types.FIL +} diff --git a/node/config/v5_to_v6.go b/node/config/v5_to_v6.go new file mode 100644 index 000000000..31df72b5d --- /dev/null +++ b/node/config/v5_to_v6.go @@ -0,0 +1,28 @@ +package config + +import ( + "fmt" +) + +// Migrate from config version 5 to version 6 +func v5Tov6(cfgPath string) (string, error) { + cfg, err := FromFile(cfgPath, DefaultBoost()) + if err != nil { + return "", fmt.Errorf("parsing config file %s: %w", cfgPath, err) + } + + boostCfg, ok := cfg.(*Boost) + if !ok { + return "", fmt.Errorf("unexpected config type %T: expected *config.Boost", cfg) + } + + // Update the Boost config version + boostCfg.ConfigVersion = 6 + + bz, err := ConfigUpdate(boostCfg, DefaultBoost(), true, false) + if err != nil { + return "", fmt.Errorf("applying configuration: %w", err) + } + + return string(bz), nil +} diff --git a/node/impl/boost.go b/node/impl/boost.go index c2a1bd853..104792426 100644 --- a/node/impl/boost.go +++ b/node/impl/boost.go @@ -7,33 +7,28 @@ import ( "fmt" "io" "net/http" - "sort" + "github.com/filecoin-project/boost/lib/legacy" "github.com/filecoin-project/boost/node/impl/backupmgr" "github.com/filecoin-project/boost/piecedirectory" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" "github.com/multiformats/go-multihash" "go.opentelemetry.io/otel/attribute" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - gfm_storagemarket "github.com/filecoin-project/boost-gfm/storagemarket" "github.com/filecoin-project/boost/api" "github.com/filecoin-project/boost/extern/boostd-data/shared/tracing" "github.com/filecoin-project/boost/gql" "github.com/filecoin-project/boost/indexprovider" "github.com/filecoin-project/boost/markets/storageadapter" - "github.com/filecoin-project/boost/node/modules/dtypes" retmarket "github.com/filecoin-project/boost/retrievalmarket/server" "github.com/filecoin-project/boost/storagemarket" "github.com/filecoin-project/boost/storagemarket/sealingpipeline" "github.com/filecoin-project/boost/storagemarket/types" - "github.com/filecoin-project/dagstore" - "github.com/filecoin-project/dagstore/shard" + "github.com/filecoin-project/go-jsonrpc/auth" lapi "github.com/filecoin-project/lotus/api" "github.com/filecoin-project/lotus/gateway" - mktsdagstore "github.com/filecoin-project/lotus/markets/dagstore" lotus_dtypes "github.com/filecoin-project/lotus/node/modules/dtypes" - "github.com/filecoin-project/lotus/storage/sectorblocks" "github.com/google/uuid" "github.com/ipfs/go-cid" logging "github.com/ipfs/go-log/v2" @@ -54,27 +49,18 @@ type BoostAPI struct { Host host.Host - DAGStore *dagstore.DAGStore - DagStoreWrapper *mktsdagstore.Wrapper - IndexBackedBlockstore dtypes.IndexBackedBlockstore // Boost StorageProvider *storagemarket.Provider IndexProvider *indexprovider.Wrapper + // Legacy Markets + LegacyDealManager legacy.LegacyDealManager + // Boost - Direct Data onboarding DirectDealsProvider *storagemarket.DirectDealsProvider - // Legacy Lotus - LegacyStorageProvider gfm_storagemarket.StorageProvider - // Lotus Markets - SectorBlocks *sectorblocks.SectorBlocks - PieceStore dtypes.ProviderPieceStore - DataTransfer dtypes.ProviderDataTransfer - - RetrievalProvider retrievalmarket.RetrievalProvider - SectorAccessor retrievalmarket.SectorAccessor - DealPublisher *storageadapter.DealPublisher + DealPublisher *storageadapter.DealPublisher // Graphsync Unpaid Retrieval GraphsyncUnpaidRetrieval *retmarket.GraphsyncUnpaidRetrieval @@ -187,16 +173,16 @@ func (sm *BoostAPI) BoostIndexerAnnounceDealRemoved(ctx context.Context, propCid return sm.IndexProvider.AnnounceBoostDealRemoved(ctx, propCid) } -func (sm *BoostAPI) BoostIndexerAnnounceDeal(ctx context.Context, deal *types.ProviderDealState) (cid.Cid, error) { - return sm.IndexProvider.AnnounceBoostDeal(ctx, deal) +func (sm *BoostAPI) BoostLegacyDealByProposalCid(ctx context.Context, propCid cid.Cid) (legacytypes.MinerDeal, error) { + return sm.LegacyDealManager.ByPropCid(propCid) } -func (sm *BoostAPI) BoostIndexerAnnounceLegacyDeal(ctx context.Context, proposalCid cid.Cid) error { - return sm.LegacyStorageProvider.AnnounceDealToIndexer(ctx, proposalCid) +func (sm *BoostAPI) BoostIndexerAnnounceDeal(ctx context.Context, deal *types.ProviderDealState) (cid.Cid, error) { + return sm.IndexProvider.AnnounceBoostDeal(ctx, deal) } -func (sm *BoostAPI) BoostLegacyDealByProposalCid(ctx context.Context, propCid cid.Cid) (gfm_storagemarket.MinerDeal, error) { - return sm.LegacyStorageProvider.GetLocalDeal(propCid) +func (sm *BoostAPI) BoostIndexerAnnounceLegacyDeal(ctx context.Context, proposalCid cid.Cid) (cid.Cid, error) { + return sm.IndexProvider.AnnounceLegcayDealToIndexer(ctx, proposalCid) } func (sm *BoostAPI) BoostOfflineDealWithData(ctx context.Context, dealUuid uuid.UUID, filePath string, delAfterImport bool) (*api.ProviderDealRejectionInfo, error) { @@ -204,347 +190,6 @@ func (sm *BoostAPI) BoostOfflineDealWithData(ctx context.Context, dealUuid uuid. return res, err } -func (sm *BoostAPI) BoostDagstoreGC(ctx context.Context) ([]api.DagstoreShardResult, error) { - if sm.DAGStore == nil { - return nil, fmt.Errorf("dagstore not available on this node") - } - - res, err := sm.DAGStore.GC(ctx) - if err != nil { - return nil, fmt.Errorf("failed to gc: %w", err) - } - - ret := make([]api.DagstoreShardResult, 0, len(res.Shards)) - for k, err := range res.Shards { - r := api.DagstoreShardResult{Key: k.String()} - if err == nil { - r.Success = true - } else { - r.Success = false - r.Error = err.Error() - } - ret = append(ret, r) - } - - return ret, nil -} - -func (sm *BoostAPI) BoostDagstoreListShards(ctx context.Context) ([]api.DagstoreShardInfo, error) { - if sm.DAGStore == nil { - return nil, fmt.Errorf("dagstore not available on this node") - } - - info := sm.DAGStore.AllShardsInfo() - ret := make([]api.DagstoreShardInfo, 0, len(info)) - for k, i := range info { - ret = append(ret, api.DagstoreShardInfo{ - Key: k.String(), - State: i.ShardState.String(), - Error: func() string { - if i.Error == nil { - return "" - } - return i.Error.Error() - }(), - }) - } - - // order by key. - sort.SliceStable(ret, func(i, j int) bool { - return ret[i].Key < ret[j].Key - }) - - return ret, nil -} - -func (sm *BoostAPI) BoostDagstorePiecesContainingMultihash(ctx context.Context, mh multihash.Multihash) ([]cid.Cid, error) { - ctx, span := tracing.Tracer.Start(ctx, "Boost.BoostDagstorePiecesContainingMultihash") - span.SetAttributes(attribute.String("multihash", mh.String())) - defer span.End() - - if sm.DAGStore == nil { - return nil, fmt.Errorf("dagstore not available on this node") - } - - ks, err := sm.DAGStore.ShardsContainingMultihash(ctx, mh) - if err != nil { - return nil, fmt.Errorf("getting pieces containing multihash %s from DAG store: %w", mh, err) - } - - pieceCids := make([]cid.Cid, 0, len(ks)) - for _, k := range ks { - pieceCid, err := cid.Parse(k.String()) - if err != nil { - return nil, fmt.Errorf("parsing DAG store shard key '%s' into cid: %w", k, err) - } - pieceCids = append(pieceCids, pieceCid) - } - - return pieceCids, nil -} - -func (sm *BoostAPI) BoostDagstoreInitializeAll(ctx context.Context, params api.DagstoreInitializeAllParams) (<-chan api.DagstoreInitializeAllEvent, error) { - if sm.DAGStore == nil { - return nil, fmt.Errorf("dagstore not available on this node") - } - - if sm.SectorAccessor == nil { - return nil, fmt.Errorf("sector accessor not available on this node") - } - - // prepare the thottler tokens. - var throttle chan struct{} - if c := params.MaxConcurrency; c > 0 { - throttle = make(chan struct{}, c) - for i := 0; i < c; i++ { - throttle <- struct{}{} - } - } - - // are we initializing only unsealed pieces? - onlyUnsealed := !params.IncludeSealed - - info := sm.DAGStore.AllShardsInfo() - var toInitialize []string - for k, i := range info { - if i.ShardState != dagstore.ShardStateNew { - continue - } - - // if we're initializing only unsealed pieces, check if there's an - // unsealed deal for this piece available. - if onlyUnsealed { - pieceCid, err := cid.Decode(k.String()) - if err != nil { - log.Warnw("DagstoreInitializeAll: failed to decode shard key as piece CID; skipping", "shard_key", k.String(), "error", err) - continue - } - - pi, err := sm.PieceStore.GetPieceInfo(pieceCid) - if err != nil { - log.Warnw("DagstoreInitializeAll: failed to get piece info; skipping", "piece_cid", pieceCid, "error", err) - continue - } - - var isUnsealed bool - for _, d := range pi.Deals { - isUnsealed, err = sm.SectorAccessor.IsUnsealed(ctx, d.SectorID, d.Offset.Unpadded(), d.Length.Unpadded()) - if err != nil { - log.Warnw("DagstoreInitializeAll: failed to get unsealed status; skipping deal", "deal_id", d.DealID, "error", err) - continue - } - if isUnsealed { - break - } - } - - if !isUnsealed { - log.Infow("DagstoreInitializeAll: skipping piece because it's sealed", "piece_cid", pieceCid, "error", err) - continue - } - } - - // yes, we're initializing this shard. - toInitialize = append(toInitialize, k.String()) - } - - total := len(toInitialize) - if total == 0 { - out := make(chan api.DagstoreInitializeAllEvent) - close(out) - return out, nil - } - - // response channel must be closed when we're done, or the context is cancelled. - // this buffering is necessary to prevent inflight children goroutines from - // publishing to a closed channel (res) when the context is cancelled. - out := make(chan api.DagstoreInitializeAllEvent, 32) // internal buffer. - res := make(chan api.DagstoreInitializeAllEvent, 32) // returned to caller. - - // pump events back to caller. - // two events per shard. - go func() { - defer close(res) - - for i := 0; i < total*2; i++ { - select { - case res <- <-out: - case <-ctx.Done(): - return - } - } - }() - - go func() { - for i, k := range toInitialize { - if throttle != nil { - select { - case <-throttle: - // acquired a throttle token, proceed. - case <-ctx.Done(): - return - } - } - - go func(k string, i int) { - r := api.DagstoreInitializeAllEvent{ - Key: k, - Event: "start", - Total: total, - Current: i + 1, // start with 1 - } - select { - case out <- r: - case <-ctx.Done(): - return - } - - err := sm.BoostDagstoreInitializeShard(ctx, k) - - if throttle != nil { - throttle <- struct{}{} - } - - r.Event = "end" - if err == nil { - r.Success = true - } else { - r.Success = false - r.Error = err.Error() - } - - select { - case out <- r: - case <-ctx.Done(): - } - }(k, i) - } - }() - - return res, nil -} - -func (sm *BoostAPI) BoostDagstoreInitializeShard(ctx context.Context, key string) error { - if sm.DAGStore == nil { - return fmt.Errorf("dagstore not available on this node") - } - - k := shard.KeyFromString(key) - - info, err := sm.DAGStore.GetShardInfo(k) - if err != nil { - return fmt.Errorf("failed to get shard info: %w", err) - } - if st := info.ShardState; st != dagstore.ShardStateNew { - return fmt.Errorf("cannot initialize shard; expected state ShardStateNew, was: %s", st.String()) - } - - ch := make(chan dagstore.ShardResult, 1) - if err = sm.DAGStore.AcquireShard(ctx, k, ch, dagstore.AcquireOpts{}); err != nil { - return fmt.Errorf("failed to acquire shard: %w", err) - } - - var res dagstore.ShardResult - select { - case res = <-ch: - case <-ctx.Done(): - return ctx.Err() - } - - if err := res.Error; err != nil { - return fmt.Errorf("failed to acquire shard: %w", err) - } - - if res.Accessor != nil { - err = res.Accessor.Close() - if err != nil { - log.Warnw("failed to close shard accessor; continuing", "shard_key", k, "error", err) - } - } - - return nil -} - -func (sm *BoostAPI) BoostDagstoreRegisterShard(ctx context.Context, key string) error { - if sm.DAGStore == nil { - return fmt.Errorf("dagstore not available on this node") - } - - // First check if the shard has already been registered - k := shard.KeyFromString(key) - _, err := sm.DAGStore.GetShardInfo(k) - if err == nil { - // Shard already registered, nothing further to do - return nil - } - // If the shard is not registered we would expect ErrShardUnknown - if !errors.Is(err, dagstore.ErrShardUnknown) { - return fmt.Errorf("getting shard info from DAG store: %w", err) - } - - pieceCid, err := cid.Parse(key) - if err != nil { - return fmt.Errorf("parsing shard key as piece cid: %w", err) - } - if err = registerShardSync(ctx, sm.DagStoreWrapper, pieceCid, "", true); err != nil { - return fmt.Errorf("failed to register shard: %w", err) - } - - return nil -} - -func (sm *BoostAPI) BoostDagstoreRecoverShard(ctx context.Context, key string) error { - if sm.DAGStore == nil { - return fmt.Errorf("dagstore not available on this node") - } - - k := shard.KeyFromString(key) - - info, err := sm.DAGStore.GetShardInfo(k) - if err != nil { - return fmt.Errorf("failed to get shard info: %w", err) - } - if st := info.ShardState; st != dagstore.ShardStateErrored { - return fmt.Errorf("cannot recover shard; expected state ShardStateErrored, was: %s", st.String()) - } - - ch := make(chan dagstore.ShardResult, 1) - if err = sm.DAGStore.RecoverShard(ctx, k, ch, dagstore.RecoverOpts{}); err != nil { - return fmt.Errorf("failed to recover shard: %w", err) - } - - var res dagstore.ShardResult - select { - case res = <-ch: - case <-ctx.Done(): - return ctx.Err() - } - - return res.Error -} - -func (sm *BoostAPI) BoostDagstoreDestroyShard(ctx context.Context, key string) error { - if sm.DAGStore == nil { - return fmt.Errorf("dagstore not available on this node") - } - - // First check if the shard has already been registered - k := shard.KeyFromString(key) - _, err := sm.DAGStore.GetShardInfo(k) - if err != nil { - return fmt.Errorf("unable to query dagstore for shard info: %w", err) - } - - pieceCid, err := cid.Parse(key) - if err != nil { - return fmt.Errorf("parsing shard key as piece cid: %w", err) - } - if err = destroyShardSync(ctx, sm.DagStoreWrapper, pieceCid); err != nil { - return fmt.Errorf("failed to destroy shard: %w", err) - } - return nil -} - func (sm *BoostAPI) BoostDirectDeal(ctx context.Context, params types.DirectDealParams) (*api.ProviderDealRejectionInfo, error) { return nil, fmt.Errorf("not implemented") // return sm.DirectDealsProvider.Import(ctx, params) @@ -578,32 +223,3 @@ func (sm *BoostAPI) PdBuildIndexForPieceCid(ctx context.Context, piececid cid.Ci func (sm *BoostAPI) OnlineBackup(ctx context.Context, dstDir string) error { return sm.Bkp.Backup(ctx, dstDir) } - -func registerShardSync(ctx context.Context, ds *mktsdagstore.Wrapper, pieceCid cid.Cid, carPath string, eagerInit bool) error { - resch := make(chan dagstore.ShardResult, 1) - if err := ds.RegisterShard(ctx, pieceCid, carPath, eagerInit, resch); err != nil { - return err - } - - select { - case <-ctx.Done(): - return ctx.Err() - case res := <-resch: - return res.Error - } -} - -func destroyShardSync(ctx context.Context, ds *mktsdagstore.Wrapper, pieceCid cid.Cid) error { - resch := make(chan dagstore.ShardResult, 1) - - if err := ds.DestroyShard(ctx, pieceCid, resch); err != nil { - return err - } - - select { - case <-ctx.Done(): - return ctx.Err() - case res := <-resch: - return res.Error - } -} diff --git a/node/impl/boost_legacy.go b/node/impl/boost_legacy.go deleted file mode 100644 index 5e4c4ce1b..000000000 --- a/node/impl/boost_legacy.go +++ /dev/null @@ -1,239 +0,0 @@ -package impl - -import ( - "context" - "fmt" - "os" - "strconv" - "time" - - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/boost-gfm/storagemarket" - "github.com/filecoin-project/boost/api" - "github.com/filecoin-project/go-address" - datatransfer "github.com/filecoin-project/go-data-transfer" - "github.com/filecoin-project/go-state-types/abi" - lapi "github.com/filecoin-project/lotus/api" - "github.com/filecoin-project/lotus/chain/types" - "github.com/ipfs/go-cid" - peer "github.com/libp2p/go-libp2p/core/peer" -) - -func (sm *BoostAPI) MarketListDataTransfers(ctx context.Context) ([]api.DataTransferChannel, error) { - inProgressChannels, err := sm.DataTransfer.InProgressChannels(ctx) - if err != nil { - return nil, err - } - - unpaidRetrievals := sm.GraphsyncUnpaidRetrieval.List() - - // Get legacy, paid retrievals - apiChannels := make([]api.DataTransferChannel, 0, len(inProgressChannels)+len(unpaidRetrievals)) - for _, channelState := range inProgressChannels { - apiChannels = append(apiChannels, api.NewDataTransferChannel(sm.Host.ID(), channelState)) - } - - // Include unpaid retrievals - for _, ur := range unpaidRetrievals { - apiChannels = append(apiChannels, api.NewDataTransferChannel(sm.Host.ID(), ur.ChannelState())) - } - - return apiChannels, nil -} - -func (sm *BoostAPI) MarketRestartDataTransfer(ctx context.Context, transferID datatransfer.TransferID, otherPeer peer.ID, isInitiator bool) error { - selfPeer := sm.Host.ID() - if isInitiator { - return sm.DataTransfer.RestartDataTransferChannel(ctx, datatransfer.ChannelID{Initiator: selfPeer, Responder: otherPeer, ID: transferID}) - } - return sm.DataTransfer.RestartDataTransferChannel(ctx, datatransfer.ChannelID{Initiator: otherPeer, Responder: selfPeer, ID: transferID}) -} - -func (sm *BoostAPI) MarketCancelDataTransfer(ctx context.Context, transferID datatransfer.TransferID, otherPeer peer.ID, isInitiator bool) error { - selfPeer := sm.Host.ID() - - // Attempt to cancel unpaid first, if that succeeds, we're done - err := sm.GraphsyncUnpaidRetrieval.CancelTransfer(ctx, transferID, &otherPeer) - if err == nil { - return nil - } - - // Legacy, paid retrievals - if isInitiator { - return sm.DataTransfer.CloseDataTransferChannel(ctx, datatransfer.ChannelID{Initiator: selfPeer, Responder: otherPeer, ID: transferID}) - } - return sm.DataTransfer.CloseDataTransferChannel(ctx, datatransfer.ChannelID{Initiator: otherPeer, Responder: selfPeer, ID: transferID}) -} - -func (sm *BoostAPI) MarketDataTransferUpdates(ctx context.Context) (<-chan api.DataTransferChannel, error) { - channels := make(chan api.DataTransferChannel) - - unsub := sm.DataTransfer.SubscribeToEvents(func(evt datatransfer.Event, channelState datatransfer.ChannelState) { - channel := api.NewDataTransferChannel(sm.Host.ID(), channelState) - select { - case <-ctx.Done(): - case channels <- channel: - } - }) - - go func() { - defer unsub() - <-ctx.Done() - }() - - return channels, nil -} - -func (sm *BoostAPI) MarketListRetrievalDeals(ctx context.Context) ([]retrievalmarket.ProviderDealState, error) { - deals := sm.RetrievalProvider.ListDeals() - unpaidRetrievals := sm.GraphsyncUnpaidRetrieval.List() - - out := make([]retrievalmarket.ProviderDealState, 0, len(deals)+len(unpaidRetrievals)) - - for _, deal := range deals { - if deal.ChannelID != nil { - if deal.ChannelID.Initiator == "" || deal.ChannelID.Responder == "" { - deal.ChannelID = nil // don't try to push unparsable peer IDs over jsonrpc - } - } - out = append(out, deal) - } - - for _, ur := range unpaidRetrievals { - out = append(out, ur.ProviderDealState()) - } - - return out, nil -} - -func (sm *BoostAPI) MarketImportDealData(ctx context.Context, propCid cid.Cid, path string) error { - fi, err := os.Open(path) - if err != nil { - return fmt.Errorf("failed to open file: %w", err) - } - defer fi.Close() //nolint:errcheck - - return sm.LegacyStorageProvider.ImportDataForDeal(ctx, propCid, fi) -} - -func (sm *BoostAPI) MarketSetRetrievalAsk(ctx context.Context, rask *retrievalmarket.Ask) error { - sm.RetrievalProvider.SetAsk(rask) - return nil -} - -func (sm *BoostAPI) MarketGetRetrievalAsk(ctx context.Context) (*retrievalmarket.Ask, error) { - return sm.RetrievalProvider.GetAsk(), nil -} - -func (sm *BoostAPI) DealsConsiderOnlineStorageDeals(ctx context.Context) (bool, error) { - return sm.ConsiderOnlineStorageDealsConfigFunc() -} - -func (sm *BoostAPI) DealsSetConsiderOnlineStorageDeals(ctx context.Context, b bool) error { - return sm.SetConsiderOnlineStorageDealsConfigFunc(b) -} - -func (sm *BoostAPI) DealsConsiderOnlineRetrievalDeals(ctx context.Context) (bool, error) { - return sm.ConsiderOnlineRetrievalDealsConfigFunc() -} - -func (sm *BoostAPI) DealsSetConsiderOnlineRetrievalDeals(ctx context.Context, b bool) error { - return sm.SetConsiderOnlineRetrievalDealsConfigFunc(b) -} - -func (sm *BoostAPI) DealsConsiderOfflineStorageDeals(ctx context.Context) (bool, error) { - return sm.ConsiderOfflineStorageDealsConfigFunc() -} - -func (sm *BoostAPI) DealsSetConsiderOfflineStorageDeals(ctx context.Context, b bool) error { - return sm.SetConsiderOfflineStorageDealsConfigFunc(b) -} - -func (sm *BoostAPI) DealsConsiderOfflineRetrievalDeals(ctx context.Context) (bool, error) { - return sm.ConsiderOfflineRetrievalDealsConfigFunc() -} - -func (sm *BoostAPI) DealsSetConsiderOfflineRetrievalDeals(ctx context.Context, b bool) error { - return sm.SetConsiderOfflineRetrievalDealsConfigFunc(b) -} - -func (sm *BoostAPI) DealsConsiderVerifiedStorageDeals(ctx context.Context) (bool, error) { - return sm.ConsiderVerifiedStorageDealsConfigFunc() -} - -func (sm *BoostAPI) DealsSetConsiderVerifiedStorageDeals(ctx context.Context, b bool) error { - return sm.SetConsiderVerifiedStorageDealsConfigFunc(b) -} - -func (sm *BoostAPI) DealsConsiderUnverifiedStorageDeals(ctx context.Context) (bool, error) { - return sm.ConsiderUnverifiedStorageDealsConfigFunc() -} - -func (sm *BoostAPI) DealsSetConsiderUnverifiedStorageDeals(ctx context.Context, b bool) error { - return sm.SetConsiderUnverifiedStorageDealsConfigFunc(b) -} - -func (sm *BoostAPI) DealsGetExpectedSealDurationFunc(ctx context.Context) (time.Duration, error) { - return sm.GetExpectedSealDurationFunc() -} - -func (sm *BoostAPI) DealsSetExpectedSealDurationFunc(ctx context.Context, d time.Duration) error { - return sm.SetExpectedSealDurationFunc(d) -} - -func (sm *BoostAPI) DealsPieceCidBlocklist(ctx context.Context) ([]cid.Cid, error) { - return sm.StorageDealPieceCidBlocklistConfigFunc() -} - -func (sm *BoostAPI) DealsSetPieceCidBlocklist(ctx context.Context, cids []cid.Cid) error { - return sm.SetStorageDealPieceCidBlocklistConfigFunc(cids) -} - -func (sm *BoostAPI) MarketSetAsk(ctx context.Context, price types.BigInt, verifiedPrice types.BigInt, duration abi.ChainEpoch, minPieceSize abi.PaddedPieceSize, maxPieceSize abi.PaddedPieceSize) error { - options := []storagemarket.StorageAskOption{ - storagemarket.MinPieceSize(minPieceSize), - storagemarket.MaxPieceSize(maxPieceSize), - } - - return sm.LegacyStorageProvider.SetAsk(price, verifiedPrice, duration, options...) -} - -func (sm *BoostAPI) MarketListIncompleteDeals(ctx context.Context) ([]storagemarket.MinerDeal, error) { - return sm.LegacyStorageProvider.ListLocalDeals() -} - -func (sm *BoostAPI) MarketGetAsk(ctx context.Context) (*storagemarket.SignedStorageAsk, error) { - return sm.LegacyStorageProvider.GetAsk(), nil -} - -func (sm *BoostAPI) ActorSectorSize(ctx context.Context, addr address.Address) (abi.SectorSize, error) { - mi, err := sm.Full.StateMinerInfo(ctx, addr, types.EmptyTSK) - if err != nil { - return 0, err - } - return mi.SectorSize, nil -} - -func (sm *BoostAPI) RuntimeSubsystems(context.Context) (res lapi.MinerSubsystems, err error) { - return []lapi.MinerSubsystem{lapi.SubsystemMarkets}, nil -} - -func (sm *BoostAPI) MarketPendingDeals(ctx context.Context) (lapi.PendingDealInfo, error) { - return sm.DealPublisher.PendingDeals(), nil -} - -func (sm *BoostAPI) SectorsRefs(ctx context.Context) (map[string][]lapi.SealedRef, error) { - // json can't handle cids as map keys - out := map[string][]lapi.SealedRef{} - - refs, err := sm.SectorBlocks.List(ctx) - if err != nil { - return nil, err - } - - for k, v := range refs { - out[strconv.FormatUint(k, 10)] = v - } - - return out, nil -} diff --git a/node/modules/client.go b/node/modules/client.go deleted file mode 100644 index 08751f981..000000000 --- a/node/modules/client.go +++ /dev/null @@ -1,176 +0,0 @@ -package modules - -import ( - "bytes" - "context" - "errors" - "fmt" - "os" - "path/filepath" - "time" - - "go.uber.org/fx" - - "github.com/filecoin-project/go-state-types/abi" - "github.com/ipfs/go-datastore" - "github.com/ipfs/go-datastore/namespace" - "github.com/libp2p/go-libp2p/core/host" - - "github.com/filecoin-project/boost-gfm/discovery" - discoveryimpl "github.com/filecoin-project/boost-gfm/discovery/impl" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - retrievalimpl "github.com/filecoin-project/boost-gfm/retrievalmarket/impl" - rmnet "github.com/filecoin-project/boost-gfm/retrievalmarket/network" - "github.com/filecoin-project/boost-gfm/storagemarket" - storageimpl "github.com/filecoin-project/boost-gfm/storagemarket/impl" - "github.com/filecoin-project/boost-gfm/storagemarket/impl/requestvalidation" - smnet "github.com/filecoin-project/boost-gfm/storagemarket/network" - "github.com/filecoin-project/boost/markets" - marketevents "github.com/filecoin-project/boost/markets/loggers" - "github.com/filecoin-project/boost/markets/retrievaladapter" - "github.com/filecoin-project/boost/markets/storageadapter" - "github.com/filecoin-project/boost/node/modules/dtypes" - "github.com/filecoin-project/lotus/blockstore" - "github.com/filecoin-project/lotus/chain/market" - "github.com/filecoin-project/lotus/journal" - "github.com/filecoin-project/lotus/node/config" - "github.com/filecoin-project/lotus/node/impl/full" - payapi "github.com/filecoin-project/lotus/node/impl/paych" - lotus_dtypes "github.com/filecoin-project/lotus/node/modules/dtypes" - "github.com/filecoin-project/lotus/node/repo" - "github.com/filecoin-project/lotus/node/repo/imports" -) - -func HandleMigrateClientFunds(lc fx.Lifecycle, ds lotus_dtypes.MetadataDS, wallet full.WalletAPI, fundMgr *market.FundManager) { - lc.Append(fx.Hook{ - OnStart: func(ctx context.Context) error { - addr, err := wallet.WalletDefaultAddress(ctx) - // nothing to be done if there is no default address - if err != nil { - return nil - } - b, err := ds.Get(ctx, datastore.NewKey("/marketfunds/client")) - if err != nil { - if errors.Is(err, datastore.ErrNotFound) { - return nil - } - log.Errorf("client funds migration - getting datastore value: %v", err) - return nil - } - - var value abi.TokenAmount - if err = value.UnmarshalCBOR(bytes.NewReader(b)); err != nil { - log.Errorf("client funds migration - unmarshalling datastore value: %v", err) - return nil - } - _, err = fundMgr.Reserve(ctx, addr, addr, value) - if err != nil { - log.Errorf("client funds migration - reserving funds (wallet %s, addr %s, funds %d): %v", - addr, addr, value, err) - return nil - } - - return ds.Delete(ctx, datastore.NewKey("/marketfunds/client")) - }, - }) -} - -func ClientImportMgr(ds lotus_dtypes.MetadataDS, r repo.LockedRepo) (lotus_dtypes.ClientImportMgr, error) { - // store the imports under the repo's `imports` subdirectory. - dir := filepath.Join(r.Path(), "imports") - if err := os.MkdirAll(dir, 0755); err != nil { - return nil, fmt.Errorf("failed to create directory %s: %w", dir, err) - } - - ns := namespace.Wrap(ds, datastore.NewKey("/client")) - return imports.NewManager(ns, dir), nil -} - -// TODO this should be removed. -func ClientBlockstore() dtypes.ClientBlockstore { - // in most cases this is now unused in normal operations -- however, it's important to preserve for the IPFS use case - return blockstore.WrapIDStore(blockstore.FromDatastore(datastore.NewMapDatastore())) -} - -// RegisterClientValidator is an initialization hook that registers the client -// request validator with the data transfer module as the validator for -// StorageDataTransferVoucher types -func RegisterClientValidator(crv dtypes.ClientRequestValidator, dtm dtypes.ClientDataTransfer) { - if err := dtm.RegisterVoucherType(&requestvalidation.StorageDataTransferVoucher{}, (*requestvalidation.UnifiedRequestValidator)(crv)); err != nil { - panic(err) - } -} - -// NewClientDatastore creates a datastore for the client to store its deals -func NewClientDatastore(ds lotus_dtypes.MetadataDS) dtypes.ClientDatastore { - return namespace.Wrap(ds, datastore.NewKey("/deals/client")) -} - -// StorageBlockstoreAccessor returns the default storage blockstore accessor -// from the import manager. -func StorageBlockstoreAccessor(importmgr dtypes.ClientImportMgr) storagemarket.BlockstoreAccessor { - return storageadapter.NewImportsBlockstoreAccessor(importmgr) -} - -// RetrievalBlockstoreAccessor returns the default retrieval blockstore accessor -// using the subdirectory `retrievals` under the repo. -func RetrievalBlockstoreAccessor(r repo.LockedRepo) (retrievalmarket.BlockstoreAccessor, error) { - dir := filepath.Join(r.Path(), "retrievals") - if err := os.MkdirAll(dir, 0755); err != nil { - return nil, fmt.Errorf("failed to create directory %s: %w", dir, err) - } - return retrievaladapter.NewCARBlockstoreAccessor(dir), nil -} - -func StorageClient(lc fx.Lifecycle, h host.Host, dataTransfer dtypes.ClientDataTransfer, discovery *discoveryimpl.Local, - deals dtypes.ClientDatastore, scn storagemarket.StorageClientNode, accessor storagemarket.BlockstoreAccessor, j journal.Journal) (storagemarket.StorageClient, error) { - // go-fil-markets protocol retries: - // 1s, 5s, 25s, 2m5s, 5m x 11 ~= 1 hour - marketsRetryParams := smnet.RetryParameters(time.Second, 5*time.Minute, 15, 5) - net := smnet.NewFromLibp2pHost(h, marketsRetryParams) - - c, err := storageimpl.NewClient(net, dataTransfer, discovery, deals, scn, accessor, storageimpl.DealPollingInterval(time.Second), storageimpl.MaxTraversalLinks(config.MaxTraversalLinks)) - if err != nil { - return nil, err - } - c.OnReady(marketevents.ReadyLogger("storage client")) - lc.Append(fx.Hook{ - OnStart: func(ctx context.Context) error { - c.SubscribeToEvents(marketevents.StorageClientLogger) - - evtType := j.RegisterEventType("markets/storage/client", "state_change") - c.SubscribeToEvents(markets.StorageClientJournaler(j, evtType)) - - return c.Start(ctx) - }, - OnStop: func(context.Context) error { - return c.Stop() - }, - }) - return c, nil -} - -// RetrievalClient creates a new retrieval client attached to the client blockstore -func RetrievalClient(lc fx.Lifecycle, h host.Host, r repo.LockedRepo, dt dtypes.ClientDataTransfer, payAPI payapi.PaychAPI, resolver discovery.PeerResolver, - ds lotus_dtypes.MetadataDS, chainAPI full.ChainAPI, stateAPI full.StateAPI, accessor retrievalmarket.BlockstoreAccessor, j journal.Journal) (retrievalmarket.RetrievalClient, error) { - - adapter := retrievaladapter.NewRetrievalClientNode(false, payAPI, chainAPI, stateAPI) - network := rmnet.NewFromLibp2pHost(h) - ds = namespace.Wrap(ds, datastore.NewKey("/retrievals/client")) - client, err := retrievalimpl.NewClient(network, dt, adapter, resolver, ds, accessor) - if err != nil { - return nil, err - } - client.OnReady(marketevents.ReadyLogger("retrieval client")) - lc.Append(fx.Hook{ - OnStart: func(ctx context.Context) error { - client.SubscribeToEvents(marketevents.RetrievalClientLogger) - - evtType := j.RegisterEventType("markets/retrieval/client", "state_change") - client.SubscribeToEvents(markets.RetrievalClientJournaler(j, evtType)) - - return client.Start(ctx) - }, - }) - return client, nil -} diff --git a/node/modules/dealfilter.go b/node/modules/dealfilter.go index d916d7989..36eb3d4ef 100644 --- a/node/modules/dealfilter.go +++ b/node/modules/dealfilter.go @@ -5,8 +5,8 @@ import ( "fmt" "time" - "github.com/filecoin-project/boost-gfm/retrievalmarket" "github.com/filecoin-project/boost/node/modules/dtypes" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" "github.com/filecoin-project/boost/storagemarket/dealfilter" "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/lotus/api/v1api" @@ -140,7 +140,7 @@ func RetrievalDealFilter(userFilter dtypes.RetrievalDealFilter) func(onlineOk dt offlineOk dtypes.ConsiderOfflineRetrievalDealsConfigFunc) dtypes.RetrievalDealFilter { return func(onlineOk dtypes.ConsiderOnlineRetrievalDealsConfigFunc, offlineOk dtypes.ConsiderOfflineRetrievalDealsConfigFunc) dtypes.RetrievalDealFilter { - return func(ctx context.Context, state retrievalmarket.ProviderDealState) (bool, string, error) { + return func(ctx context.Context, state legacyretrievaltypes.ProviderDealState) (bool, string, error) { b, err := onlineOk() if err != nil { return false, "miner error", err diff --git a/node/modules/directdeals.go b/node/modules/directdeals.go index 0bca1759a..146e01968 100644 --- a/node/modules/directdeals.go +++ b/node/modules/directdeals.go @@ -3,7 +3,6 @@ package modules import ( "database/sql" - gfm_storagemarket "github.com/filecoin-project/boost-gfm/storagemarket" "github.com/filecoin-project/boost/db" "github.com/filecoin-project/boost/fundmanager" "github.com/filecoin-project/boost/indexprovider" @@ -24,13 +23,12 @@ import ( "go.uber.org/fx" ) -func NewDirectDealsProvider(provAddr address.Address, cfg *config.Boost) func(lc fx.Lifecycle, h host.Host, fullnodeApi v1api.FullNode, sqldb *sql.DB, directDealsDB *db.DirectDealsDB, fundMgr *fundmanager.FundManager, storageMgr *storagemanager.StorageManager, dp *storageadapter.DealPublisher, secb *sectorblocks.SectorBlocks, commpc types.CommpCalculator, commpt storagemarket.CommpThrottle, sps sealingpipeline.API, df dtypes.StorageDealFilter, logsSqlDB *LogSqlDB, logsDB *db.LogsDB, piecedirectory *piecedirectory.PieceDirectory, ip *indexprovider.Wrapper, lp gfm_storagemarket.StorageProvider, cdm *storagemarket.ChainDealManager) (*storagemarket.DirectDealsProvider, error) { +func NewDirectDealsProvider(provAddr address.Address, cfg *config.Boost) func(lc fx.Lifecycle, h host.Host, fullnodeApi v1api.FullNode, sqldb *sql.DB, directDealsDB *db.DirectDealsDB, fundMgr *fundmanager.FundManager, storageMgr *storagemanager.StorageManager, dp *storageadapter.DealPublisher, secb *sectorblocks.SectorBlocks, commpc types.CommpCalculator, commpt storagemarket.CommpThrottle, sps sealingpipeline.API, df dtypes.StorageDealFilter, logsSqlDB *LogSqlDB, logsDB *db.LogsDB, piecedirectory *piecedirectory.PieceDirectory, ip *indexprovider.Wrapper, cdm *storagemarket.ChainDealManager) (*storagemarket.DirectDealsProvider, error) { return func(lc fx.Lifecycle, h host.Host, fullnodeApi v1api.FullNode, sqldb *sql.DB, directDealsDB *db.DirectDealsDB, fundMgr *fundmanager.FundManager, storageMgr *storagemanager.StorageManager, dp *storageadapter.DealPublisher, secb *sectorblocks.SectorBlocks, commpc types.CommpCalculator, commpt storagemarket.CommpThrottle, sps sealingpipeline.API, df dtypes.StorageDealFilter, logsSqlDB *LogSqlDB, logsDB *db.LogsDB, - piecedirectory *piecedirectory.PieceDirectory, ip *indexprovider.Wrapper, - lp gfm_storagemarket.StorageProvider, cdm *storagemarket.ChainDealManager) (*storagemarket.DirectDealsProvider, error) { + piecedirectory *piecedirectory.PieceDirectory, ip *indexprovider.Wrapper, cdm *storagemarket.ChainDealManager) (*storagemarket.DirectDealsProvider, error) { dl := logs.NewDealLogger(logsDB) diff --git a/node/modules/dtypes/miner.go b/node/modules/dtypes/miner.go index 99ed3e6a0..248093888 100644 --- a/node/modules/dtypes/miner.go +++ b/node/modules/dtypes/miner.go @@ -2,10 +2,11 @@ package dtypes import ( "context" - "github.com/ipfs/go-cid" "time" - "github.com/filecoin-project/boost-gfm/retrievalmarket" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" + "github.com/ipfs/go-cid" + "github.com/filecoin-project/boost/storagemarket/dealfilter" "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/lotus/storage/pipeline/sealiface" @@ -92,4 +93,4 @@ type GetMaxDealStartDelayFunc func() (time.Duration, error) type StorageDealFilter dealfilter.StorageDealFilter type RetrievalDealFilter dealfilter.RetrievalDealFilter -type RetrievalPricingFunc func(ctx context.Context, dealPricingParams retrievalmarket.PricingInput) (retrievalmarket.Ask, error) +type RetrievalPricingFunc func(ctx context.Context, dealPricingParams legacyretrievaltypes.PricingInput) (legacyretrievaltypes.Ask, error) diff --git a/node/modules/dtypes/storage.go b/node/modules/dtypes/storage.go index 901b76197..b117621af 100644 --- a/node/modules/dtypes/storage.go +++ b/node/modules/dtypes/storage.go @@ -2,15 +2,13 @@ package dtypes import ( graphsync "github.com/filecoin-project/boost-graphsync" - datatransfer "github.com/filecoin-project/go-data-transfer" - dtnet "github.com/filecoin-project/go-data-transfer/network" + "github.com/filecoin-project/boost/datatransfer" + dtnet "github.com/filecoin-project/boost/datatransfer/network" "github.com/filecoin-project/go-statestore" bserv "github.com/ipfs/boxo/blockservice" - exchange "github.com/ipfs/boxo/exchange" + "github.com/ipfs/boxo/exchange" "github.com/ipfs/go-datastore" - "github.com/filecoin-project/boost-gfm/piecestore" - "github.com/filecoin-project/boost-gfm/storagemarket/impl/requestvalidation" ipfsblockstore "github.com/ipfs/boxo/blockstore" "github.com/filecoin-project/lotus/blockstore" @@ -72,21 +70,14 @@ type ChainBlockService bserv.BlockService type ClientImportMgr *imports.Manager type ClientBlockstore blockstore.BasicBlockstore type ClientDealStore *statestore.StateStore -type ClientRequestValidator *requestvalidation.UnifiedRequestValidator type ClientDatastore datastore.Batching type Graphsync graphsync.GraphExchange -// ClientDataTransfer is a data transfer manager for the client -type ClientDataTransfer datatransfer.Manager -type ProviderDataTransfer datatransfer.Manager type ProviderTransferNetwork dtnet.DataTransferNetwork type ProviderTransport datatransfer.Transport type ProviderDealStore *statestore.StateStore -type ProviderPieceStore piecestore.PieceStore - -type ProviderRequestValidator *requestvalidation.UnifiedRequestValidator type StagingBlockstore blockstore.BasicBlockstore type StagingGraphsync graphsync.GraphExchange diff --git a/node/modules/graphsync.go b/node/modules/graphsync.go index 6f9e76b0f..5785dc9f4 100644 --- a/node/modules/graphsync.go +++ b/node/modules/graphsync.go @@ -5,8 +5,6 @@ import ( "sync" "time" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - retrievalimpl "github.com/filecoin-project/boost-gfm/retrievalmarket/impl" graphsync "github.com/filecoin-project/boost-graphsync/impl" gsnet "github.com/filecoin-project/boost-graphsync/network" "github.com/filecoin-project/boost-graphsync/storeutil" @@ -16,75 +14,23 @@ import ( "github.com/filecoin-project/boost/node/modules/dtypes" "github.com/filecoin-project/boost/piecedirectory" "github.com/filecoin-project/boost/retrievalmarket/server" - "github.com/filecoin-project/go-state-types/abi" + retrievalimpl "github.com/filecoin-project/boost/retrievalmarket/server" "github.com/filecoin-project/lotus/metrics" lotus_helpers "github.com/filecoin-project/lotus/node/modules/helpers" "github.com/ipfs/kubo/core/node/helpers" - "github.com/ipld/go-ipld-prime" - provider "github.com/ipni/index-provider" - "github.com/ipni/index-provider/engine" "github.com/libp2p/go-libp2p/core/host" "go.opencensus.io/stats" "go.uber.org/fx" ) -var _ server.AskGetter = (*ProxyAskGetter)(nil) - -// ProxyAskGetter is used to avoid circular dependencies: -// RetrievalProvider depends on RetrievalGraphsync, which depends on RetrievalProvider's -// GetAsk method. -// We create an AskGetter that returns zero-priced asks by default. -// Then we set the AskGetter to the RetrievalProvider after it's been created. -type ProxyAskGetter struct { - server.AskGetter -} - -func (ag *ProxyAskGetter) GetAsk() *retrievalmarket.Ask { - if ag.AskGetter == nil { - return &retrievalmarket.Ask{ - PricePerByte: abi.NewTokenAmount(0), - UnsealPrice: abi.NewTokenAmount(0), - } - } - return ag.AskGetter.GetAsk() -} - -func NewAskGetter() *ProxyAskGetter { - return &ProxyAskGetter{} -} - -func SetAskGetter(proxy *ProxyAskGetter, rp retrievalmarket.RetrievalProvider) { - proxy.AskGetter = rp -} - -// LinkSystemProv is used to avoid circular dependencies -type LinkSystemProv struct { - *ipld.LinkSystem -} - -func NewLinkSystemProvider() *LinkSystemProv { - return &LinkSystemProv{} -} - -func (p *LinkSystemProv) LinkSys() *ipld.LinkSystem { - return p.LinkSystem -} - -func SetLinkSystem(proxy *LinkSystemProv, prov provider.Interface) { - e, ok := prov.(*engine.Engine) - if ok { - proxy.LinkSystem = e.LinkSystem() - } -} - // RetrievalGraphsync creates a graphsync instance used to serve retrievals. -func RetrievalGraphsync(parallelTransfersForStorage uint64, parallelTransfersForStoragePerPeer uint64, parallelTransfersForRetrieval uint64) func(mctx lotus_helpers.MetricsCtx, lc fx.Lifecycle, pid *piecedirectory.PieceDirectory, h host.Host, net dtypes.ProviderTransferNetwork, dealDecider dtypes.RetrievalDealFilter, pstore dtypes.ProviderPieceStore, sa *lib.MultiMinerAccessor, askGetter server.AskGetter, ls server.LinkSystemProvider) (*server.GraphsyncUnpaidRetrieval, error) { - return func(mctx lotus_helpers.MetricsCtx, lc fx.Lifecycle, pid *piecedirectory.PieceDirectory, h host.Host, net dtypes.ProviderTransferNetwork, dealDecider dtypes.RetrievalDealFilter, pstore dtypes.ProviderPieceStore, sa *lib.MultiMinerAccessor, askGetter server.AskGetter, ls server.LinkSystemProvider) (*server.GraphsyncUnpaidRetrieval, error) { +func RetrievalGraphsync(parallelTransfersForRetrieval uint64) func(mctx lotus_helpers.MetricsCtx, lc fx.Lifecycle, pid *piecedirectory.PieceDirectory, h host.Host, net dtypes.ProviderTransferNetwork, dealDecider dtypes.RetrievalDealFilter, sa *lib.MultiMinerAccessor, askGetter server.RetrievalAskGetter) (*server.GraphsyncUnpaidRetrieval, error) { + return func(mctx lotus_helpers.MetricsCtx, lc fx.Lifecycle, pid *piecedirectory.PieceDirectory, h host.Host, net dtypes.ProviderTransferNetwork, dealDecider dtypes.RetrievalDealFilter, sa *lib.MultiMinerAccessor, askGetter server.RetrievalAskGetter) (*server.GraphsyncUnpaidRetrieval, error) { // Graphsync tracks metrics separately, pass nil blockMetrics to the remote blockstore rb := remoteblockstore.NewRemoteBlockstore(pid, nil) // Create a Graphsync instance - mkgs := Graphsync(parallelTransfersForStorage, parallelTransfersForStoragePerPeer, parallelTransfersForRetrieval) + mkgs := Graphsync(parallelTransfersForRetrieval) gs := mkgs(mctx, lc, rb, pid, h) // Wrap the Graphsync instance with a handler for unpaid retrieval requests @@ -94,7 +40,8 @@ func RetrievalGraphsync(parallelTransfersForStorage uint64, parallelTransfersFor SectorAccessor: sa, AskStore: askGetter, } - gsupr, err := server.NewGraphsyncUnpaidRetrieval(h.ID(), gs, net, vdeps, ls) + + gsupr, err := server.NewGraphsyncUnpaidRetrieval(h.ID(), gs, net, vdeps) if err != nil { return nil, err } @@ -115,7 +62,7 @@ func RetrievalGraphsync(parallelTransfersForStorage uint64, parallelTransfersFor } } -func Graphsync(parallelTransfersForStorage uint64, parallelTransfersForStoragePerPeer uint64, parallelTransfersForRetrieval uint64) func(mctx lotus_helpers.MetricsCtx, lc fx.Lifecycle, ibs dtypes.StagingBlockstore, pid *piecedirectory.PieceDirectory, h host.Host) dtypes.StagingGraphsync { +func Graphsync(parallelTransfersForRetrieval uint64) func(mctx lotus_helpers.MetricsCtx, lc fx.Lifecycle, ibs dtypes.StagingBlockstore, pid *piecedirectory.PieceDirectory, h host.Host) dtypes.StagingGraphsync { return func(mctx lotus_helpers.MetricsCtx, lc fx.Lifecycle, ibs dtypes.StagingBlockstore, pid *piecedirectory.PieceDirectory, h host.Host) dtypes.StagingGraphsync { graphsyncNetwork := gsnet.NewFromLibp2pHost(h) lsys := storeutil.LinkSystemForBlockstore(ibs) @@ -123,9 +70,7 @@ func Graphsync(parallelTransfersForStorage uint64, parallelTransfersForStoragePe graphsyncNetwork, lsys, graphsync.RejectAllRequestsByDefault(), - graphsync.MaxInProgressIncomingRequests(parallelTransfersForRetrieval), - graphsync.MaxInProgressIncomingRequestsPerPeer(parallelTransfersForStoragePerPeer), - graphsync.MaxInProgressOutgoingRequests(parallelTransfersForStorage), + graphsync.MaxInProgressOutgoingRequests(parallelTransfersForRetrieval), graphsync.MaxLinksPerIncomingRequests(config.MaxTraversalLinks), graphsync.MaxLinksPerOutgoingRequests(config.MaxTraversalLinks)) diff --git a/node/modules/legacy_markets.go b/node/modules/legacy_markets.go deleted file mode 100644 index 9cffb5bf0..000000000 --- a/node/modules/legacy_markets.go +++ /dev/null @@ -1,127 +0,0 @@ -package modules - -import ( - "context" - "fmt" - "os" - "path/filepath" - - piecefilestore "github.com/filecoin-project/boost-gfm/filestore" - "github.com/filecoin-project/boost-gfm/storagemarket" - storageimpl "github.com/filecoin-project/boost-gfm/storagemarket/impl" - "github.com/filecoin-project/boost-gfm/storagemarket/impl/storedask" - smnet "github.com/filecoin-project/boost-gfm/storagemarket/network" - "github.com/filecoin-project/boost-gfm/stores" - "github.com/filecoin-project/boost/markets/idxprov" - "github.com/filecoin-project/boost/node/config" - "github.com/filecoin-project/boost/node/modules/dtypes" - "github.com/filecoin-project/go-address" - datatransferv2 "github.com/filecoin-project/go-data-transfer/v2" - lotus_gfm_filestore "github.com/filecoin-project/go-fil-markets/filestore" - lotus_gfm_storagemarket "github.com/filecoin-project/go-fil-markets/storagemarket" - lotus_modules "github.com/filecoin-project/lotus/node/modules" - lotus_dtypes "github.com/filecoin-project/lotus/node/modules/dtypes" - "github.com/filecoin-project/lotus/node/repo" - "github.com/ipfs/go-datastore" - "github.com/ipfs/go-datastore/namespace" - provider "github.com/ipni/index-provider" - "github.com/libp2p/go-libp2p/core/host" - "github.com/libp2p/go-libp2p/core/protocol" -) - -func StorageProvider(minerAddress lotus_dtypes.MinerAddress, - storedAsk *storedask.StoredAsk, - h host.Host, ds lotus_dtypes.MetadataDS, - r repo.LockedRepo, - pieceStore dtypes.ProviderPieceStore, - indexer provider.Interface, - dataTransfer dtypes.ProviderDataTransfer, - spn storagemarket.StorageProviderNode, - df storageimpl.DealDeciderFunc, - dsw stores.DAGStoreWrapper, - meshCreator idxprov.MeshCreator, cfg config.DealmakingConfig, -) (storagemarket.StorageProvider, error) { - var opts []smnet.Option - - // Provide an empty deal protocol list to the libp2p host if legacy deals are disabled - // These protocols are handled by Boost provider and all legacy deals are rejected - if !cfg.EnableLegacyStorageDeals { - opts = append(opts, smnet.SupportedDealProtocols([]protocol.ID{})) - } - - net := smnet.NewFromLibp2pHost(h, opts...) - - dir := filepath.Join(r.Path(), lotus_modules.StagingAreaDirName) - err := os.MkdirAll(dir, os.ModePerm) - if err != nil { - return nil, fmt.Errorf("creating directory for staging legacy markets deals %s: %w", dir, err) - } - - store, err := piecefilestore.NewLocalFileStore(piecefilestore.OsPath(dir)) - if err != nil { - return nil, err - } - - opt := storageimpl.CustomDealDecisionLogic(df) - - return storageimpl.NewProvider( - net, - namespace.Wrap(ds, datastore.NewKey("/deals/provider")), - store, - dsw, - indexer, - pieceStore, - dataTransfer, - spn, - address.Address(minerAddress), - storedAsk, - meshCreator, - opt, - ) -} - -func DealDeciderFn(df lotus_dtypes.StorageDealFilter) storageimpl.DealDeciderFunc { - return func(ctx context.Context, deal storagemarket.MinerDeal) (bool, string, error) { - return df(ctx, toLotusGFMMinerDeal(deal)) - } -} - -func toLotusGFMMinerDeal(deal storagemarket.MinerDeal) lotus_gfm_storagemarket.MinerDeal { - lotusGFMDeal := lotus_gfm_storagemarket.MinerDeal{ - ClientDealProposal: deal.ClientDealProposal, - ProposalCid: deal.ProposalCid, - AddFundsCid: deal.AddFundsCid, - PublishCid: deal.PublishCid, - Miner: deal.Miner, - Client: deal.Client, - State: deal.State, - PiecePath: lotus_gfm_filestore.Path(deal.PiecePath), - MetadataPath: lotus_gfm_filestore.Path(deal.MetadataPath), - SlashEpoch: deal.SlashEpoch, - FastRetrieval: deal.FastRetrieval, - Message: deal.Message, - FundsReserved: deal.FundsReserved, - AvailableForRetrieval: deal.AvailableForRetrieval, - DealID: deal.DealID, - CreationTime: deal.CreationTime, - SectorNumber: deal.SectorNumber, - InboundCAR: deal.InboundCAR, - } - if deal.Ref != nil { - lotusGFMDeal.Ref = &lotus_gfm_storagemarket.DataRef{ - TransferType: deal.Ref.TransferType, - Root: deal.Ref.Root, - PieceCid: deal.Ref.PieceCid, - PieceSize: deal.Ref.PieceSize, - RawBlockSize: deal.Ref.RawBlockSize, - } - } - if deal.TransferChannelId != nil { - lotusGFMDeal.TransferChannelId = &datatransferv2.ChannelID{ - Initiator: deal.TransferChannelId.Initiator, - Responder: deal.TransferChannelId.Responder, - ID: datatransferv2.TransferID(deal.TransferChannelId.ID), - } - } - return lotusGFMDeal -} diff --git a/node/modules/piecedirectory.go b/node/modules/piecedirectory.go index 846e3b3a0..3e5fdfa99 100644 --- a/node/modules/piecedirectory.go +++ b/node/modules/piecedirectory.go @@ -5,21 +5,13 @@ import ( "fmt" "time" - "github.com/filecoin-project/boost-gfm/piecestore" - "github.com/filecoin-project/boost-gfm/shared" - "github.com/filecoin-project/boost-gfm/storagemarket" - "github.com/filecoin-project/boost-gfm/stores" "github.com/filecoin-project/boost/cmd/lib" bdclient "github.com/filecoin-project/boost/extern/boostd-data/client" - "github.com/filecoin-project/boost/extern/boostd-data/model" "github.com/filecoin-project/boost/extern/boostd-data/svc" "github.com/filecoin-project/boost/extern/boostd-data/yugabyte" - "github.com/filecoin-project/boost/gql" "github.com/filecoin-project/boost/node/config" "github.com/filecoin-project/boost/piecedirectory" "github.com/filecoin-project/boost/sectorstatemgr" - "github.com/filecoin-project/dagstore" - "github.com/filecoin-project/dagstore/shard" "github.com/filecoin-project/go-address" "github.com/filecoin-project/go-jsonrpc" "github.com/filecoin-project/lotus/api" @@ -27,10 +19,6 @@ import ( "github.com/filecoin-project/lotus/node/modules/dtypes" lotus_dtypes "github.com/filecoin-project/lotus/node/modules/dtypes" lotus_repo "github.com/filecoin-project/lotus/node/repo" - bstore "github.com/ipfs/boxo/blockstore" - blocks "github.com/ipfs/go-block-format" - "github.com/ipfs/go-cid" - carindex "github.com/ipld/go-car/v2/index" "go.uber.org/fx" ) @@ -119,7 +107,7 @@ func NewMultiminerSectorAccessor(cfg *config.Boost) func(full v1api.FullNode) *l return func(full v1api.FullNode) *lib.MultiMinerAccessor { // Get the endpoints of all the miners that this boost node can query // for retrieval data when serving graphsync retrievals - storageApiInfos := cfg.Dealmaking.GraphsyncStorageAccessApiInfo + storageApiInfos := cfg.Retrievals.Graphsync.GraphsyncStorageAccessApiInfo if len(storageApiInfos) == 0 { // If the endpoints aren't explicitly configured, fall back to just // serving retrieval data from the same endpoint where data is stored to @@ -159,10 +147,6 @@ func NewPieceDirectory(cfg *config.Boost) func(lc fx.Lifecycle, maddr dtypes.Min } } -func NewPieceStore(pm *piecedirectory.PieceDirectory, maddr address.Address) piecestore.PieceStore { - return &boostPieceStoreWrapper{piecedirectory: pm, maddr: maddr} -} - func NewPieceDoctor(cfg *config.Boost) func(lc fx.Lifecycle, maddr lotus_dtypes.MinerAddress, store *bdclient.Store, ssm *sectorstatemgr.SectorStateMgr, fullnodeApi api.FullNode) *piecedirectory.Doctor { return func(lc fx.Lifecycle, maddr lotus_dtypes.MinerAddress, store *bdclient.Store, ssm *sectorstatemgr.SectorStateMgr, fullnodeApi api.FullNode) *piecedirectory.Doctor { if !cfg.LocalIndexDirectory.EnablePieceDoctor { @@ -183,158 +167,3 @@ func NewPieceDoctor(cfg *config.Boost) func(lc fx.Lifecycle, maddr lotus_dtypes. return doc } } - -type boostPieceStoreWrapper struct { - piecedirectory *piecedirectory.PieceDirectory - maddr address.Address -} - -func (pw *boostPieceStoreWrapper) Start(ctx context.Context) error { - return nil -} - -func (pw *boostPieceStoreWrapper) OnReady(ready shared.ReadyFunc) { - go ready(nil) -} - -func (pw *boostPieceStoreWrapper) AddDealForPiece(pieceCID cid.Cid, proposalCid cid.Cid, dealInfo piecestore.DealInfo) error { - di := model.DealInfo{ - DealUuid: proposalCid.String(), - IsLegacy: true, - ChainDealID: dealInfo.DealID, - MinerAddr: pw.maddr, - SectorID: dealInfo.SectorID, - PieceOffset: dealInfo.Offset, - PieceLength: dealInfo.Length, - // TODO: It would be nice if there's some way to figure out the CAR - // file size here (but I don't think there is an easy way in legacy - // markets without having access to the piece data itself) - CarLength: 0, - IsDirectDeal: false, // There should be no direct deals from legacy code base - } - return pw.piecedirectory.AddDealForPiece(context.Background(), pieceCID, di) -} - -func (pw *boostPieceStoreWrapper) AddPieceBlockLocations(pieceCID cid.Cid, blockLocations map[cid.Cid]piecestore.BlockLocation) error { - // This method is no longer needed, we keep the CAR file index in the piece metadata store - return nil -} - -func (pw *boostPieceStoreWrapper) GetPieceInfo(pieceCID cid.Cid) (piecestore.PieceInfo, error) { - pieceDeals, err := pw.piecedirectory.GetPieceDeals(context.TODO(), pieceCID) - if err != nil { - return piecestore.PieceInfo{}, fmt.Errorf("getting piece deals from piece metadata store: %w", err) - } - - dis := make([]piecestore.DealInfo, 0, len(pieceDeals)) - for _, pd := range pieceDeals { - dis = append(dis, piecestore.DealInfo{ - DealID: pd.ChainDealID, - SectorID: pd.SectorID, - Offset: pd.PieceOffset, - Length: pd.PieceLength, - }) - } - return piecestore.PieceInfo{ - PieceCID: pieceCID, - Deals: dis, - }, nil -} - -func (pw *boostPieceStoreWrapper) GetCIDInfo(payloadCID cid.Cid) (piecestore.CIDInfo, error) { - // This is no longer used (CLI calls piece metadata store instead) - return piecestore.CIDInfo{}, nil -} - -func (pw *boostPieceStoreWrapper) ListCidInfoKeys() ([]cid.Cid, error) { - // This is no longer used (CLI calls piece metadata store instead) - return nil, nil -} - -func (pw *boostPieceStoreWrapper) ListPieceInfoKeys() ([]cid.Cid, error) { - // This is no longer used (CLI calls piece metadata store instead) - return nil, nil -} - -func NewDAGStoreWrapper(pm *piecedirectory.PieceDirectory) stores.DAGStoreWrapper { - // TODO: lotus_modules.NewStorageMarketProvider and lotus_modules.RetrievalProvider - // take a concrete *dagstore.Wrapper as a parameter. Create boost versions of these - // that instead take a stores.DAGStoreWrapper parameter - return &boostDAGStoreWrapper{piecedirectory: pm} -} - -type boostDAGStoreWrapper struct { - piecedirectory *piecedirectory.PieceDirectory -} - -func (dw *boostDAGStoreWrapper) DestroyShard(ctx context.Context, pieceCid cid.Cid, resch chan dagstore.ShardResult) error { - // This is no longer used (CLI calls piece metadata store instead) - return nil -} - -// Legacy markets calls piecestore.AddDealForPiece before RegisterShard, -// so we do the real work in AddDealForPiece. -func (dw *boostDAGStoreWrapper) RegisterShard(ctx context.Context, pieceCid cid.Cid, carPath string, eagerInit bool, resch chan dagstore.ShardResult) error { - res := dagstore.ShardResult{ - Key: shard.KeyFromCID(pieceCid), - Error: nil, - Accessor: nil, - } - - select { - case resch <- res: - return nil - case <-ctx.Done(): - return ctx.Err() - } -} - -func (dw *boostDAGStoreWrapper) LoadShard(ctx context.Context, pieceCid cid.Cid) (stores.ClosableBlockstore, error) { - bs, err := dw.piecedirectory.GetBlockstore(ctx, pieceCid) - if err != nil { - return nil, fmt.Errorf("getting blockstore in LoadShard: %w", err) - } - return closableBlockstore{Blockstore: bs}, nil -} - -func (dw *boostDAGStoreWrapper) MigrateDeals(ctx context.Context, deals []storagemarket.MinerDeal) (bool, error) { - // MigrateDeals is no longer needed - it's handled by the piece metadata store - return false, nil -} - -func (dw *boostDAGStoreWrapper) GetPiecesContainingBlock(blockCID cid.Cid) ([]cid.Cid, error) { - return dw.piecedirectory.PiecesContainingMultihash(context.TODO(), blockCID.Hash()) -} - -func (dw *boostDAGStoreWrapper) GetIterableIndexForPiece(pieceCid cid.Cid) (carindex.IterableIndex, error) { - return dw.piecedirectory.GetIterableIndex(context.TODO(), pieceCid) -} - -func (dw *boostDAGStoreWrapper) Close() error { - return nil -} - -type closableBlockstore struct { - bstore.Blockstore -} - -func (c closableBlockstore) Close() error { - return nil -} - -func NewBlockGetter(pd *piecedirectory.PieceDirectory) gql.BlockGetter { - return &pdBlockGetter{pd: pd} -} - -type pdBlockGetter struct { - pd *piecedirectory.PieceDirectory -} - -func (p *pdBlockGetter) Get(ctx context.Context, c cid.Cid) (blocks.Block, error) { - bz, err := p.pd.BlockstoreGet(ctx, c) - if err != nil { - return nil, err - } - - return blocks.NewBlockWithCid(bz, c) -} diff --git a/node/modules/provider_piece_store.go b/node/modules/provider_piece_store.go deleted file mode 100644 index d8fa71c74..000000000 --- a/node/modules/provider_piece_store.go +++ /dev/null @@ -1,25 +0,0 @@ -package modules - -import ( - "context" - "github.com/filecoin-project/boost/node/modules/dtypes" - - marketevents "github.com/filecoin-project/boost/markets/loggers" - "github.com/filecoin-project/boost/piecedirectory" - "github.com/filecoin-project/go-address" - lotus_dtypes "github.com/filecoin-project/lotus/node/modules/dtypes" - "go.uber.org/fx" -) - -// NewProviderPieceStore creates a statestore for storing metadata about pieces -// shared by the storage and retrieval providers -func NewProviderPieceStore(lc fx.Lifecycle, pm *piecedirectory.PieceDirectory, maddr lotus_dtypes.MinerAddress) (dtypes.ProviderPieceStore, error) { - ps := NewPieceStore(pm, address.Address(maddr)) - ps.OnReady(marketevents.ReadyLogger("piecestore")) - lc.Append(fx.Hook{ - OnStart: func(ctx context.Context) error { - return ps.Start(ctx) - }, - }) - return ps, nil -} diff --git a/node/modules/retrieval.go b/node/modules/retrieval.go index 89a7f681e..22ea50cca 100644 --- a/node/modules/retrieval.go +++ b/node/modules/retrieval.go @@ -7,11 +7,9 @@ import ( "path" "time" - lotus_retrievalmarket "github.com/filecoin-project/boost-gfm/retrievalmarket" "github.com/filecoin-project/boost/cmd/booster-bitswap/bitswap" "github.com/filecoin-project/boost/db" "github.com/filecoin-project/boost/node/config" - "github.com/filecoin-project/boost/node/modules/dtypes" "github.com/filecoin-project/boost/protocolproxy" "github.com/filecoin-project/boost/retrievalmarket/lp2pimpl" "github.com/filecoin-project/boost/retrievalmarket/rtvllog" @@ -29,7 +27,7 @@ import ( // based off the host and boost config func bitswapMultiAddrs(cfg *config.Boost, h host.Host) ([]multiaddr.Multiaddr, error) { // if BitswapPublicAddresses is empty, that means we'll be serving bitswap directly from this host, so just return host multiaddresses - if len(cfg.Dealmaking.BitswapPublicAddresses) == 0 { + if len(cfg.Retrievals.Bitswap.BitswapPublicAddresses) == 0 { maddr, err := peer.AddrInfoToP2pAddrs(&peer.AddrInfo{ ID: h.ID(), Addrs: h.Addrs(), @@ -45,7 +43,7 @@ func bitswapMultiAddrs(cfg *config.Boost, h host.Host) ([]multiaddr.Multiaddr, e // parse all of the public multiaddrs var addrs []multiaddr.Multiaddr - for _, addrString := range cfg.Dealmaking.BitswapPublicAddresses { + for _, addrString := range cfg.Retrievals.Bitswap.BitswapPublicAddresses { addr, err := multiaddr.NewMultiaddr(addrString) if err != nil { return nil, fmt.Errorf("Could not parse bitswap address '%s' as multiaddr: %w", addrString, err) @@ -54,9 +52,9 @@ func bitswapMultiAddrs(cfg *config.Boost, h host.Host) ([]multiaddr.Multiaddr, e } // in order to make these multiaddrs fully dialable, we encapsulate the bitswap peer id inside of them - bsPeerID, err := peer.Decode(cfg.Dealmaking.BitswapPeerID) + bsPeerID, err := peer.Decode(cfg.Retrievals.Bitswap.BitswapPeerID) if err != nil { - return nil, fmt.Errorf("Could not parse bitswap peer id '%s': %w", cfg.Dealmaking.BitswapPeerID, err) + return nil, fmt.Errorf("Could not parse bitswap peer id '%s': %w", cfg.Retrievals.Bitswap.BitswapPeerID, err) } return peer.AddrInfoToP2pAddrs(&peer.AddrInfo{ ID: bsPeerID, @@ -85,12 +83,12 @@ func NewTransportsListener(cfg *config.Boost) func(h host.Host) (*lp2pimpl.Trans // If there's an http retrieval address specified, add HTTP to the list // of supported protocols - if cfg.Dealmaking.HTTPRetrievalMultiaddr != "" { - maddr, err := multiaddr.NewMultiaddr(cfg.Dealmaking.HTTPRetrievalMultiaddr) + if cfg.Retrievals.HTTP.HTTPRetrievalMultiaddr != "" { + maddr, err := multiaddr.NewMultiaddr(cfg.Retrievals.HTTP.HTTPRetrievalMultiaddr) if err != nil { msg := "HTTPRetrievalURL must be in multi-address format. " msg += "Could not parse '%s' as multiaddr: %w" - return nil, fmt.Errorf(msg, cfg.Dealmaking.HTTPRetrievalMultiaddr, err) + return nil, fmt.Errorf(msg, cfg.Retrievals.HTTP.HTTPRetrievalMultiaddr, err) } protos = append(protos, types.Protocol{ Name: "http", @@ -100,7 +98,7 @@ func NewTransportsListener(cfg *config.Boost) func(h host.Host) (*lp2pimpl.Trans // If there's a bitswap peer address specified, add bitswap to the list // of supported protocols - if cfg.Dealmaking.BitswapPeerID != "" { + if cfg.Retrievals.Bitswap.BitswapPeerID != "" { addrs, err := bitswapMultiAddrs(cfg, h) if err != nil { return nil, err @@ -156,19 +154,15 @@ func NewRetrievalLogDB(db *RetrievalSqlDB) *rtvllog.RetrievalLogDB { } // Write graphsync retrieval updates to the database -func HandleRetrievalGraphsyncUpdates(duration time.Duration, stalledDuration time.Duration) func(lc fx.Lifecycle, db *rtvllog.RetrievalLogDB, m lotus_retrievalmarket.RetrievalProvider, dt dtypes.ProviderDataTransfer, gsur *server.GraphsyncUnpaidRetrieval) { - return func(lc fx.Lifecycle, db *rtvllog.RetrievalLogDB, m lotus_retrievalmarket.RetrievalProvider, dt dtypes.ProviderDataTransfer, gsur *server.GraphsyncUnpaidRetrieval) { - rel := rtvllog.NewRetrievalLog(db, duration, dt, stalledDuration, gsur) +func HandleRetrievalGraphsyncUpdates(duration time.Duration, stalledDuration time.Duration) func(lc fx.Lifecycle, db *rtvllog.RetrievalLogDB, gsur *server.GraphsyncUnpaidRetrieval) { + return func(lc fx.Lifecycle, db *rtvllog.RetrievalLogDB, gsur *server.GraphsyncUnpaidRetrieval) { + rel := rtvllog.NewRetrievalLog(db, duration, stalledDuration, gsur) relctx, cancel := context.WithCancel(context.Background()) type unsubFn func() var unsubs []unsubFn lc.Append(fx.Hook{ OnStart: func(ctx context.Context) error { - unsubs = append(unsubs, unsubFn(m.SubscribeToEvents(rel.OnRetrievalEvent))) - unsubs = append(unsubs, unsubFn(m.SubscribeToQueryEvents(rel.OnQueryEvent))) - unsubs = append(unsubs, unsubFn(m.SubscribeToValidationEvents(rel.OnValidationEvent))) - unsubs = append(unsubs, unsubFn(dt.SubscribeToEvents(rel.OnDataTransferEvent))) unsubs = append(unsubs, unsubFn(gsur.SubscribeToDataTransferEvents(rel.OnDataTransferEvent))) unsubs = append(unsubs, unsubFn(gsur.SubscribeToMarketsEvents(rel.OnRetrievalEvent))) rel.Start(relctx) @@ -189,8 +183,8 @@ func NewProtocolProxy(cfg *config.Boost) func(h host.Host) (*protocolproxy.Proto return func(h host.Host) (*protocolproxy.ProtocolProxy, error) { peerConfig := map[peer.ID][]protocol.ID{} // add bitswap if a peer id is set AND the peer is only private - if cfg.Dealmaking.BitswapPeerID != "" && len(cfg.Dealmaking.BitswapPublicAddresses) == 0 { - bsPeerID, err := peer.Decode(cfg.Dealmaking.BitswapPeerID) + if cfg.Retrievals.Bitswap.BitswapPeerID != "" && len(cfg.Retrievals.Bitswap.BitswapPublicAddresses) == 0 { + bsPeerID, err := peer.Decode(cfg.Retrievals.Bitswap.BitswapPeerID) if err != nil { return nil, err } diff --git a/node/modules/storageminer.go b/node/modules/storageminer.go index 9303db0f2..bdab69cbd 100644 --- a/node/modules/storageminer.go +++ b/node/modules/storageminer.go @@ -9,63 +9,44 @@ import ( "path" "time" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - retrievalimpl "github.com/filecoin-project/boost-gfm/retrievalmarket/impl" - rmnet "github.com/filecoin-project/boost-gfm/retrievalmarket/network" - "github.com/filecoin-project/boost-gfm/shared" - gfm_storagemarket "github.com/filecoin-project/boost-gfm/storagemarket" - storageimpl "github.com/filecoin-project/boost-gfm/storagemarket/impl" - "github.com/filecoin-project/boost-gfm/storagemarket/impl/storedask" - "github.com/filecoin-project/boost-gfm/stores" "github.com/filecoin-project/boost/cmd/lib" + dtnet "github.com/filecoin-project/boost/datatransfer/network" + dtgstransport "github.com/filecoin-project/boost/datatransfer/transport/graphsync" "github.com/filecoin-project/boost/db" "github.com/filecoin-project/boost/extern/boostd-data/shared/tracing" "github.com/filecoin-project/boost/fundmanager" - "github.com/filecoin-project/boost/gql" "github.com/filecoin-project/boost/indexprovider" "github.com/filecoin-project/boost/lib/legacy" "github.com/filecoin-project/boost/lib/mpoolmonitor" - "github.com/filecoin-project/boost/markets/idxprov" - marketevents "github.com/filecoin-project/boost/markets/loggers" - "github.com/filecoin-project/boost/markets/pricing" "github.com/filecoin-project/boost/markets/sectoraccessor" "github.com/filecoin-project/boost/markets/storageadapter" "github.com/filecoin-project/boost/node/config" "github.com/filecoin-project/boost/node/impl/backupmgr" "github.com/filecoin-project/boost/node/modules/dtypes" "github.com/filecoin-project/boost/piecedirectory" - brm "github.com/filecoin-project/boost/retrievalmarket/lib" - "github.com/filecoin-project/boost/retrievalmarket/rtvllog" "github.com/filecoin-project/boost/retrievalmarket/server" - "github.com/filecoin-project/boost/sectorstatemgr" "github.com/filecoin-project/boost/storagemanager" "github.com/filecoin-project/boost/storagemarket" "github.com/filecoin-project/boost/storagemarket/logs" "github.com/filecoin-project/boost/storagemarket/lp2pimpl" "github.com/filecoin-project/boost/storagemarket/sealingpipeline" + "github.com/filecoin-project/boost/storagemarket/storedask" "github.com/filecoin-project/boost/storagemarket/types" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" "github.com/filecoin-project/boost/transport/httptransport" - "github.com/filecoin-project/dagstore" - "github.com/filecoin-project/dagstore/indexbs" - "github.com/filecoin-project/dagstore/shard" "github.com/filecoin-project/go-address" - dtnet "github.com/filecoin-project/go-data-transfer/network" - dtgstransport "github.com/filecoin-project/go-data-transfer/transport/graphsync" - lotus_gfm_shared "github.com/filecoin-project/go-fil-markets/shared" - lotus_gfm_storagemarket "github.com/filecoin-project/go-fil-markets/storagemarket" - "github.com/filecoin-project/go-state-types/abi" - "github.com/filecoin-project/go-state-types/big" + vfsm "github.com/filecoin-project/go-ds-versioning/pkg/fsm" "github.com/filecoin-project/go-state-types/builtin" - "github.com/filecoin-project/go-state-types/builtin/v8/verifreg" "github.com/filecoin-project/go-state-types/builtin/v9/account" - "github.com/filecoin-project/go-state-types/builtin/v9/market" "github.com/filecoin-project/go-state-types/crypto" "github.com/filecoin-project/go-state-types/exitcode" + "github.com/filecoin-project/go-statemachine/fsm" "github.com/filecoin-project/lotus/api/v1api" "github.com/filecoin-project/lotus/build" ctypes "github.com/filecoin-project/lotus/chain/types" ltypes "github.com/filecoin-project/lotus/chain/types" "github.com/filecoin-project/lotus/gateway" + "github.com/filecoin-project/lotus/lib/backupds" "github.com/filecoin-project/lotus/lib/sigs" lotus_dtypes "github.com/filecoin-project/lotus/node/modules/dtypes" "github.com/filecoin-project/lotus/node/modules/helpers" @@ -75,8 +56,6 @@ import ( "github.com/ipfs/go-cid" "github.com/ipfs/go-datastore" "github.com/ipfs/go-datastore/namespace" - "github.com/ipni/go-libipni/metadata" - provider "github.com/ipni/index-provider" "github.com/libp2p/go-libp2p/core/host" "go.uber.org/fx" "go.uber.org/multierr" @@ -280,14 +259,6 @@ func mutateCfg(r lotus_repo.LockedRepo, mutator func(*config.Boost)) error { return multierr.Combine(typeErr, setConfigErr) } -func StorageNetworkName(ctx helpers.MetricsCtx, a v1api.FullNode) (dtypes.NetworkName, error) { - n, err := a.StateNetworkName(ctx) - if err != nil { - return "", err - } - return dtypes.NetworkName(n), nil -} - func NewBoostDB(r lotus_repo.LockedRepo) (*sql.DB, error) { // fixes error "database is locked", caused by concurrent access from deal goroutines to a single sqlite3 db connection // see: https://github.com/mattn/go-sqlite3#:~:text=Error%3A%20database%20is%20locked @@ -330,21 +301,7 @@ func NewFundsDB(sqldb *sql.DB) *db.FundsDB { return db.NewFundsDB(sqldb) } -func HandleRetrieval(host host.Host, lc fx.Lifecycle, m retrievalmarket.RetrievalProvider) { - m.OnReady(marketevents.ReadyLogger("retrieval provider")) - lc.Append(fx.Hook{ - - OnStart: func(ctx context.Context) error { - m.SubscribeToEvents(marketevents.RetrievalProviderLogger) - return m.Start(ctx) - }, - OnStop: func(context.Context) error { - return m.Stop() - }, - }) -} - -func HandleQueryAsk(lc fx.Lifecycle, h host.Host, maddr lotus_dtypes.MinerAddress, pd *piecedirectory.PieceDirectory, sa *lib.MultiMinerAccessor, askStore server.AskGetter, full v1api.FullNode) { +func HandleQueryAsk(lc fx.Lifecycle, h host.Host, maddr lotus_dtypes.MinerAddress, pd *piecedirectory.PieceDirectory, sa *lib.MultiMinerAccessor, askStore server.RetrievalAskGetter, full v1api.FullNode) { handler := server.NewQueryAskHandler(h, address.Address(maddr), pd, sa, askStore, full) lc.Append(fx.Hook{ OnStart: func(ctx context.Context) error { @@ -362,47 +319,16 @@ func NewSectorStateDB(sqldb *sql.DB) *db.SectorStateDB { return db.NewSectorStateDB(sqldb) } -func HandleLegacyDeals(mctx helpers.MetricsCtx, lc fx.Lifecycle, lsp gfm_storagemarket.StorageProvider) error { - log.Info("starting legacy storage provider") - ctx := helpers.LifecycleCtx(mctx, lc) - lsp.OnReady(marketevents.ReadyLogger("storage provider")) - lc.Append(fx.Hook{ - OnStart: func(context.Context) error { - lsp.SubscribeToEvents(marketevents.StorageProviderLogger) - return lsp.Start(ctx) - }, - OnStop: func(context.Context) error { - return lsp.Stop() - }, - }) - return nil -} - -func HandleBoostLibp2pDeals(cfg *config.Boost) func(lc fx.Lifecycle, h host.Host, prov *storagemarket.Provider, ddprov *storagemarket.DirectDealsProvider, a v1api.FullNode, legacySP gfm_storagemarket.StorageProvider, idxProv *indexprovider.Wrapper, plDB *db.ProposalLogsDB, spApi sealingpipeline.API) { - return func(lc fx.Lifecycle, h host.Host, prov *storagemarket.Provider, ddprov *storagemarket.DirectDealsProvider, a v1api.FullNode, legacySP gfm_storagemarket.StorageProvider, idxProv *indexprovider.Wrapper, plDB *db.ProposalLogsDB, spApi sealingpipeline.API) { +func HandleBoostLibp2pDeals(cfg *config.Boost) func(lc fx.Lifecycle, h host.Host, prov *storagemarket.Provider, a v1api.FullNode, idxProv *indexprovider.Wrapper, plDB *db.ProposalLogsDB, spApi sealingpipeline.API) { + return func(lc fx.Lifecycle, h host.Host, prov *storagemarket.Provider, a v1api.FullNode, idxProv *indexprovider.Wrapper, plDB *db.ProposalLogsDB, spApi sealingpipeline.API) { - lp2pnet := lp2pimpl.NewDealProvider(h, prov, a, plDB, spApi, cfg.Dealmaking.EnableLegacyStorageDeals) + lp2pnet := lp2pimpl.NewDealProvider(h, prov, a, plDB, spApi) lc.Append(fx.Hook{ OnStart: func(ctx context.Context) error { - // Wait for the legacy SP to fire the "ready" event before starting - // the boost SP. - // Boost overrides some listeners so it must start after the legacy SP. - errch := make(chan error, 1) - log.Info("waiting for legacy storage provider 'ready' event") - legacySP.OnReady(func(err error) { - errch <- err - }) - err := <-errch - if err != nil { - log.Errorf("failed to start legacy storage provider: %w", err) - return err - } - log.Info("legacy storage provider started successfully") - // Start the Boost SP log.Info("starting boost storage provider") - err = prov.Start() + err := prov.Start() if err != nil { return fmt.Errorf("starting storage provider: %w", err) } @@ -415,14 +341,6 @@ func HandleBoostLibp2pDeals(cfg *config.Boost) func(lc fx.Lifecycle, h host.Host log.Info("starting boost index provider wrapper") idxProv.Start(ctx) log.Info("boost index provider wrapper started successfully") - - // Start the Boost Direct Deals provider - log.Info("starting boost direct deals provider") - err = ddprov.Start(ctx) - if err != nil { - return fmt.Errorf("starting direct deals provider: %w", err) - } - log.Info("boost direct deals provider started successfully") return nil }, OnStop: func(ctx context.Context) error { @@ -531,9 +449,9 @@ func NewChainDealManager(a v1api.FullNode) *storagemarket.ChainDealManager { return storagemarket.NewChainDealManager(a, cdmCfg) } -func NewLegacyDealsManager(lc fx.Lifecycle, legacyProv gfm_storagemarket.StorageProvider) *legacy.LegacyDealsManager { +func NewLegacyDealsManager(lc fx.Lifecycle, legacyFSM fsm.Group) legacy.LegacyDealManager { ctx, cancel := context.WithCancel(context.Background()) - mgr := legacy.NewLegacyDealsManager(legacyProv) + mgr := legacy.NewLegacyDealsManager(legacyFSM) lc.Append(fx.Hook{ OnStart: func(_ context.Context) error { go mgr.Run(ctx) @@ -547,96 +465,20 @@ func NewLegacyDealsManager(lc fx.Lifecycle, legacyProv gfm_storagemarket.Storage return mgr } -func NewStorageAsk(ctx helpers.MetricsCtx, fapi v1api.FullNode, ds lotus_dtypes.MetadataDS, minerAddress lotus_dtypes.MinerAddress, spn gfm_storagemarket.StorageProviderNode) (*storedask.StoredAsk, error) { - mi, err := fapi.StateMinerInfo(ctx, address.Address(minerAddress), ltypes.EmptyTSK) - if err != nil { - return nil, err - } - - providerDs := namespace.Wrap(ds, datastore.NewKey("/deals/provider")) - // legacy this was mistake where this key was place -- so we move the legacy key if need be - err = shared.MoveKey(providerDs, "/latest-ask", "/storage-ask/latest") - if err != nil { - return nil, err - } - return storedask.NewStoredAsk(namespace.Wrap(providerDs, datastore.NewKey("/storage-ask")), datastore.NewKey("latest"), spn, address.Address(minerAddress), - gfm_storagemarket.MaxPieceSize(abi.PaddedPieceSize(mi.SectorSize))) -} - -// NewLegacyStorageProvider wraps lotus's storage provider function but additionally sets up the metadata announcement -// for legacy deals based off of Boost's configured protocols -func NewLegacyStorageProvider(cfg *config.Boost) func(minerAddress lotus_dtypes.MinerAddress, - storedAsk *storedask.StoredAsk, - h host.Host, ds lotus_dtypes.MetadataDS, - r repo.LockedRepo, - pieceStore dtypes.ProviderPieceStore, - indexer provider.Interface, - dataTransfer dtypes.ProviderDataTransfer, - spn gfm_storagemarket.StorageProviderNode, - df storageimpl.DealDeciderFunc, - dsw stores.DAGStoreWrapper, - meshCreator idxprov.MeshCreator, -) (gfm_storagemarket.StorageProvider, error) { - return func(minerAddress lotus_dtypes.MinerAddress, - storedAsk *storedask.StoredAsk, - h host.Host, ds lotus_dtypes.MetadataDS, - r repo.LockedRepo, - pieceStore dtypes.ProviderPieceStore, - indexer provider.Interface, - dataTransfer dtypes.ProviderDataTransfer, - spn gfm_storagemarket.StorageProviderNode, - df storageimpl.DealDeciderFunc, - dsw stores.DAGStoreWrapper, - meshCreator idxprov.MeshCreator, - ) (gfm_storagemarket.StorageProvider, error) { - prov, err := StorageProvider(minerAddress, storedAsk, h, ds, r, pieceStore, indexer, dataTransfer, spn, df, dsw, meshCreator, cfg.Dealmaking) - if err != nil { - return prov, err - } - p := prov.(*storageimpl.Provider) - p.Configure(storageimpl.CustomMetadataGenerator(func(deal gfm_storagemarket.MinerDeal) metadata.Metadata { - - // Announce deal to network Indexer - protocols := []metadata.Protocol{ - &metadata.GraphsyncFilecoinV1{ - PieceCID: deal.Proposal.PieceCID, - FastRetrieval: deal.FastRetrieval, - VerifiedDeal: deal.Proposal.VerifiedDeal, - }, - } - - return metadata.Default.New(protocols...) - - })) - return p, nil - } -} - -func NewCommpThrottle(cfg *config.Boost) func() storagemarket.CommpThrottle { - return func() storagemarket.CommpThrottle { - size := uint64(1) - if cfg.Dealmaking.MaxConcurrentLocalCommp > 1 { - size = cfg.Dealmaking.MaxConcurrentLocalCommp - } - return make(chan struct{}, size) - } -} - -func NewStorageMarketProvider(provAddr address.Address, cfg *config.Boost) func(lc fx.Lifecycle, h host.Host, a v1api.FullNode, sqldb *sql.DB, dealsDB *db.DealsDB, fundMgr *fundmanager.FundManager, storageMgr *storagemanager.StorageManager, dp *storageadapter.DealPublisher, secb *sectorblocks.SectorBlocks, commpc types.CommpCalculator, commpt storagemarket.CommpThrottle, sps sealingpipeline.API, df dtypes.StorageDealFilter, logsSqlDB *LogSqlDB, logsDB *db.LogsDB, piecedirectory *piecedirectory.PieceDirectory, ip *indexprovider.Wrapper, lp gfm_storagemarket.StorageProvider, cdm *storagemarket.ChainDealManager) (*storagemarket.Provider, error) { +func NewStorageMarketProvider(provAddr address.Address, cfg *config.Boost) func(lc fx.Lifecycle, h host.Host, a v1api.FullNode, sqldb *sql.DB, dealsDB *db.DealsDB, fundMgr *fundmanager.FundManager, storageMgr *storagemanager.StorageManager, sask storedask.StoredAsk, dp *storageadapter.DealPublisher, secb *sectorblocks.SectorBlocks, commpc types.CommpCalculator, commpt storagemarket.CommpThrottle, sps sealingpipeline.API, df dtypes.StorageDealFilter, logsSqlDB *LogSqlDB, logsDB *db.LogsDB, piecedirectory *piecedirectory.PieceDirectory, ip *indexprovider.Wrapper, cdm *storagemarket.ChainDealManager) (*storagemarket.Provider, error) { return func(lc fx.Lifecycle, h host.Host, a v1api.FullNode, sqldb *sql.DB, dealsDB *db.DealsDB, - fundMgr *fundmanager.FundManager, storageMgr *storagemanager.StorageManager, dp *storageadapter.DealPublisher, secb *sectorblocks.SectorBlocks, + fundMgr *fundmanager.FundManager, storageMgr *storagemanager.StorageManager, sask storedask.StoredAsk, dp *storageadapter.DealPublisher, secb *sectorblocks.SectorBlocks, commpc types.CommpCalculator, commpt storagemarket.CommpThrottle, sps sealingpipeline.API, df dtypes.StorageDealFilter, logsSqlDB *LogSqlDB, logsDB *db.LogsDB, - piecedirectory *piecedirectory.PieceDirectory, ip *indexprovider.Wrapper, - lp gfm_storagemarket.StorageProvider, cdm *storagemarket.ChainDealManager) (*storagemarket.Provider, error) { + piecedirectory *piecedirectory.PieceDirectory, ip *indexprovider.Wrapper, cdm *storagemarket.ChainDealManager) (*storagemarket.Provider, error) { prvCfg := storagemarket.Config{ MaxTransferDuration: time.Duration(cfg.Dealmaking.MaxTransferDuration), RemoteCommp: cfg.Dealmaking.RemoteCommp, TransferLimiter: storagemarket.TransferLimiterConfig{ - MaxConcurrent: cfg.Dealmaking.HttpTransferMaxConcurrentDownloads, - StallCheckPeriod: time.Duration(cfg.Dealmaking.HttpTransferStallCheckPeriod), - StallTimeout: time.Duration(cfg.Dealmaking.HttpTransferStallTimeout), + MaxConcurrent: cfg.HttpDownload.HttpTransferMaxConcurrentDownloads, + StallCheckPeriod: time.Duration(cfg.HttpDownload.HttpTransferStallCheckPeriod), + StallTimeout: time.Duration(cfg.HttpDownload.HttpTransferStallTimeout), }, DealLogDurationDays: cfg.Dealmaking.DealLogDurationDays, StorageFilter: cfg.Dealmaking.Filter, @@ -644,7 +486,8 @@ func NewStorageMarketProvider(provAddr address.Address, cfg *config.Boost) func( } dl := logs.NewDealLogger(logsDB) tspt := httptransport.New(h, dl, httptransport.NChunksOpt(cfg.HttpDownload.NChunks), httptransport.AllowPrivateIPsOpt(cfg.HttpDownload.AllowPrivateIPs)) - prov, err := storagemarket.NewProvider(prvCfg, sqldb, dealsDB, fundMgr, storageMgr, a, dp, provAddr, secb, commpc, commpt, sps, cdm, df, logsSqlDB.db, logsDB, piecedirectory, ip, lp, &signatureVerifier{a}, dl, tspt) + prov, err := storagemarket.NewProvider(prvCfg, sqldb, dealsDB, fundMgr, storageMgr, a, dp, provAddr, secb, commpc, commpt, + sps, cdm, df, logsSqlDB.db, logsDB, piecedirectory, ip, sask, &signatureVerifier{a}, dl, tspt) if err != nil { return nil, err } @@ -653,27 +496,13 @@ func NewStorageMarketProvider(provAddr address.Address, cfg *config.Boost) func( } } -func NewGraphqlServer(cfg *config.Boost) func(lc fx.Lifecycle, r repo.LockedRepo, h host.Host, prov *storagemarket.Provider, ddProv *storagemarket.DirectDealsProvider, dealsDB *db.DealsDB, directDealsDB *db.DirectDealsDB, logsDB *db.LogsDB, retDB *rtvllog.RetrievalLogDB, plDB *db.ProposalLogsDB, fundsDB *db.FundsDB, fundMgr *fundmanager.FundManager, storageMgr *storagemanager.StorageManager, publisher *storageadapter.DealPublisher, spApi sealingpipeline.API, legacyDeals *legacy.LegacyDealsManager, legacyProv gfm_storagemarket.StorageProvider, legacyDT dtypes.ProviderDataTransfer, ps dtypes.ProviderPieceStore, piecedirectory *piecedirectory.PieceDirectory, indexProv provider.Interface, idxProvWrapper *indexprovider.Wrapper, fullNode v1api.FullNode, bg gql.BlockGetter, ssm *sectorstatemgr.SectorStateMgr, mpool *mpoolmonitor.MpoolMonitor, mma *lib.MultiMinerAccessor) *gql.Server { - return func(lc fx.Lifecycle, r repo.LockedRepo, h host.Host, prov *storagemarket.Provider, ddProv *storagemarket.DirectDealsProvider, dealsDB *db.DealsDB, directDealsDB *db.DirectDealsDB, logsDB *db.LogsDB, retDB *rtvllog.RetrievalLogDB, plDB *db.ProposalLogsDB, fundsDB *db.FundsDB, fundMgr *fundmanager.FundManager, - storageMgr *storagemanager.StorageManager, publisher *storageadapter.DealPublisher, spApi sealingpipeline.API, - legacyDeals *legacy.LegacyDealsManager, legacyProv gfm_storagemarket.StorageProvider, legacyDT dtypes.ProviderDataTransfer, - ps dtypes.ProviderPieceStore, piecedirectory *piecedirectory.PieceDirectory, - indexProv provider.Interface, idxProvWrapper *indexprovider.Wrapper, fullNode v1api.FullNode, bg gql.BlockGetter, - ssm *sectorstatemgr.SectorStateMgr, mpool *mpoolmonitor.MpoolMonitor, mma *lib.MultiMinerAccessor) *gql.Server { - - resolverCtx, cancel := context.WithCancel(context.Background()) - resolver := gql.NewResolver(resolverCtx, cfg, r, h, dealsDB, directDealsDB, logsDB, retDB, plDB, fundsDB, fundMgr, storageMgr, spApi, prov, ddProv, legacyDeals, legacyProv, legacyDT, ps, piecedirectory, publisher, indexProv, idxProvWrapper, fullNode, ssm, mpool, mma) - server := gql.NewServer(cfg, resolver, bg) - - lc.Append(fx.Hook{ - OnStart: server.Start, - OnStop: func(ctx context.Context) error { - cancel() - return server.Stop(ctx) - }, - }) - - return server +func NewCommpThrottle(cfg *config.Boost) func() storagemarket.CommpThrottle { + return func() storagemarket.CommpThrottle { + size := uint64(1) + if cfg.Dealmaking.MaxConcurrentLocalCommp > 1 { + size = cfg.Dealmaking.MaxConcurrentLocalCommp + } + return make(chan struct{}, size) } } @@ -685,74 +514,6 @@ func NewSectorAccessor(cfg *config.Boost) sectoraccessor.SectorAccessorConstruct return sectoraccessor.NewCachingSectorAccessor(maxCacheSize, time.Duration(cfg.Dealmaking.IsUnsealedCacheExpiry)) } -// ShardSelector helps to resolve a circular dependency: -// The IndexBackedBlockstore has a shard selector, which needs to query the -// RetrievalProviderNode's ask to find out if it's free to retrieve a -// particular piece. -// However the RetrievalProviderNode depends on the DAGStore which depends on -// IndexBackedBlockstore. -// So we -// - create a ShardSelector that has no dependencies with a default shard -// selection function that just selects no shards -// - later call SetShardSelectorFunc to create a real shard selector function -// with all its dependencies, and set it on the ShardSelector object. -type ShardSelector struct { - Proxy indexbs.ShardSelectorF - Target indexbs.ShardSelectorF -} - -func NewShardSelector() *ShardSelector { - ss := &ShardSelector{ - // The default target function always selects no shards - Target: func(c cid.Cid, shards []shard.Key) (shard.Key, error) { - return shard.Key{}, indexbs.ErrNoShardSelected - }, - } - ss.Proxy = func(c cid.Cid, shards []shard.Key) (shard.Key, error) { - return ss.Target(c, shards) - } - - return ss -} - -func SetShardSelectorFunc(lc fx.Lifecycle, shardSelector *ShardSelector, ps dtypes.ProviderPieceStore, sa retrievalmarket.SectorAccessor, rp retrievalmarket.RetrievalProvider) error { - ctx, cancel := context.WithCancel(context.Background()) - lc.Append(fx.Hook{ - OnStop: func(ctx context.Context) error { - cancel() - return nil - }, - }) - - ss, err := brm.NewShardSelector(ctx, ps, sa, rp) - if err != nil { - return fmt.Errorf("creating shard selector: %w", err) - } - - shardSelector.Target = ss.ShardSelectorF - - return nil -} - -func NewIndexBackedBlockstore(cfg *config.Boost) func(lc fx.Lifecycle, dagst dagstore.Interface, ss *ShardSelector) (dtypes.IndexBackedBlockstore, error) { - return func(lc fx.Lifecycle, dagst dagstore.Interface, ss *ShardSelector) (dtypes.IndexBackedBlockstore, error) { - ctx, cancel := context.WithCancel(context.Background()) - lc.Append(fx.Hook{ - OnStop: func(ctx context.Context) error { - cancel() - return nil - }, - }) - - ibsds := brm.NewIndexBackedBlockstoreDagstore(dagst) - rbs, err := indexbs.NewIndexBackedBlockstore(ctx, ibsds, ss.Proxy, cfg.Dealmaking.BlockstoreCacheMaxShards, time.Duration(cfg.Dealmaking.BlockstoreCacheExpiry)) - if err != nil { - return nil, fmt.Errorf("failed to create index backed blockstore: %w", err) - } - return dtypes.IndexBackedBlockstore(rbs), nil - } -} - func NewTracing(cfg *config.Boost) func(lc fx.Lifecycle) (*tracing.Tracing, error) { return func(lc fx.Lifecycle) (*tracing.Tracing, error) { if cfg.Tracing.Enabled { @@ -787,123 +548,6 @@ func NewProviderTransport(h host.Host, gs dtypes.StagingGraphsync) dtypes.Provid return dtgstransport.NewTransport(h.ID(), gs) } -func RetrievalNetwork(h host.Host) rmnet.RetrievalMarketNetwork { - return rmnet.NewFromLibp2pHost(h) -} - -// RetrievalPricingFunc configures the pricing function to use for retrieval deals. -func RetrievalPricingFunc(cfg config.DealmakingConfig) func(_ dtypes.ConsiderOnlineRetrievalDealsConfigFunc, - _ dtypes.ConsiderOfflineRetrievalDealsConfigFunc) dtypes.RetrievalPricingFunc { - - return func(_ dtypes.ConsiderOnlineRetrievalDealsConfigFunc, - _ dtypes.ConsiderOfflineRetrievalDealsConfigFunc) dtypes.RetrievalPricingFunc { - if cfg.RetrievalPricing.Strategy == config.RetrievalPricingExternalMode { - return pricing.ExternalRetrievalPricingFunc(cfg.RetrievalPricing.External.Path) - } - - return retrievalimpl.DefaultPricingFunc(cfg.RetrievalPricing.Default.VerifiedDealsFreeTransfer) - } -} - -// RetrievalProvider creates a new retrieval provider attached to the provider blockstore -func RetrievalProvider( - maddr lotus_dtypes.MinerAddress, - adapter retrievalmarket.RetrievalProviderNode, - sa retrievalmarket.SectorAccessor, - netwk rmnet.RetrievalMarketNetwork, - ds lotus_dtypes.MetadataDS, - pieceStore dtypes.ProviderPieceStore, - dt dtypes.ProviderDataTransfer, - pricingFnc dtypes.RetrievalPricingFunc, - userFilter dtypes.RetrievalDealFilter, - dagStore stores.DAGStoreWrapper, -) (retrievalmarket.RetrievalProvider, error) { - opt := retrievalimpl.DealDeciderOpt(retrievalimpl.DealDecider(userFilter)) - - retrievalmarket.DefaultPricePerByte = big.Zero() // todo: for whatever reason this is a global var in markets - - return retrievalimpl.NewProvider( - address.Address(maddr), - adapter, - sa, - netwk, - pieceStore, - dagStore, - dt, - namespace.Wrap(ds, datastore.NewKey("/retrievals/provider")), - retrievalimpl.RetrievalPricingFunc(pricingFnc), - opt, - ) -} - -func LotusGFMStorageProviderNode(spn gfm_storagemarket.StorageProviderNode) lotus_gfm_storagemarket.StorageProviderNode { - return &lotusGFMSPN{StorageProviderNode: spn} -} - -type lotusGFMSPN struct { - gfm_storagemarket.StorageProviderNode -} - -func (l *lotusGFMSPN) GetChainHead(ctx context.Context) (lotus_gfm_shared.TipSetToken, abi.ChainEpoch, error) { - tst, ce, err := l.StorageProviderNode.GetChainHead(ctx) - return lotus_gfm_shared.TipSetToken(tst), ce, err -} - -func (l *lotusGFMSPN) GetBalance(ctx context.Context, addr address.Address, tok lotus_gfm_shared.TipSetToken) (lotus_gfm_storagemarket.Balance, error) { - //TODO implement me - panic("implement me") -} - -func (l *lotusGFMSPN) VerifySignature(ctx context.Context, signature crypto.Signature, signer address.Address, plaintext []byte, tok lotus_gfm_shared.TipSetToken) (bool, error) { - //TODO implement me - panic("implement me") -} - -func (l *lotusGFMSPN) OnDealSectorPreCommitted(ctx context.Context, provider address.Address, dealID abi.DealID, proposal market.DealProposal, publishCid *cid.Cid, cb lotus_gfm_storagemarket.DealSectorPreCommittedCallback) error { - //TODO implement me - panic("implement me") -} - -func (l *lotusGFMSPN) OnDealSectorCommitted(ctx context.Context, provider address.Address, dealID abi.DealID, sectorNumber abi.SectorNumber, proposal market.DealProposal, publishCid *cid.Cid, cb lotus_gfm_storagemarket.DealSectorCommittedCallback) error { - //TODO implement me - panic("implement me") -} - -func (l *lotusGFMSPN) OnDealExpiredOrSlashed(ctx context.Context, dealID abi.DealID, onDealExpired lotus_gfm_storagemarket.DealExpiredCallback, onDealSlashed lotus_gfm_storagemarket.DealSlashedCallback) error { - //TODO implement me - panic("implement me") -} - -func (l *lotusGFMSPN) PublishDeals(ctx context.Context, deal lotus_gfm_storagemarket.MinerDeal) (cid.Cid, error) { - //TODO implement me - panic("implement me") -} - -func (l *lotusGFMSPN) WaitForPublishDeals(ctx context.Context, mcid cid.Cid, proposal market.DealProposal) (*lotus_gfm_storagemarket.PublishDealsWaitResult, error) { - //TODO implement me - panic("implement me") -} - -func (l *lotusGFMSPN) OnDealComplete(ctx context.Context, deal lotus_gfm_storagemarket.MinerDeal, pieceSize abi.UnpaddedPieceSize, pieceReader lotus_gfm_shared.ReadSeekStarter) (*lotus_gfm_storagemarket.PackingResult, error) { - //TODO implement me - panic("implement me") -} - -func (l *lotusGFMSPN) GetMinerWorkerAddress(ctx context.Context, addr address.Address, tok lotus_gfm_shared.TipSetToken) (address.Address, error) { - //TODO implement me - panic("implement me") -} - -func (l *lotusGFMSPN) GetDataCap(ctx context.Context, addr address.Address, tok lotus_gfm_shared.TipSetToken) (*verifreg.DataCap, error) { - //TODO implement me - panic("implement me") -} - -func (l *lotusGFMSPN) GetProofType(ctx context.Context, addr address.Address, tok lotus_gfm_shared.TipSetToken) (abi.RegisteredSealProof, error) { - //TODO implement me - panic("implement me") -} - func NewMpoolMonitor(cfg *config.Boost) func(lc fx.Lifecycle, a v1api.FullNode) *mpoolmonitor.MpoolMonitor { return func(lc fx.Lifecycle, a v1api.FullNode) *mpoolmonitor.MpoolMonitor { mpm := mpoolmonitor.NewMonitor(a, cfg.Monitoring.MpoolAlertEpochs) @@ -916,3 +560,29 @@ func NewMpoolMonitor(cfg *config.Boost) func(lc fx.Lifecycle, a v1api.FullNode) return mpm } } + +func NewLegacyDealsFSM(cfg *config.Boost) func(lc fx.Lifecycle, mds lotus_dtypes.MetadataDS) (fsm.Group, error) { + return func(lc fx.Lifecycle, mds lotus_dtypes.MetadataDS) (fsm.Group, error) { + // Get the deals FSM + bds, err := backupds.Wrap(mds, "") + if err != nil { + return nil, fmt.Errorf("opening backupds: %w", err) + } + provDS := namespace.Wrap(bds, datastore.NewKey("/deals/provider")) + deals, migrate, err := vfsm.NewVersionedFSM(provDS, fsm.Parameters{ + StateType: legacytypes.MinerDeal{}, + StateKeyField: "State", + }, nil, "2") + if err != nil { + return nil, fmt.Errorf("reading legacy deals from datastore: %w", err) + } + ctx := context.Background() + + err = migrate(ctx) + if err != nil { + return nil, fmt.Errorf("running provider fsm migration script: %w", err) + } + + return deals, err + } +} diff --git a/node/modules/storageminer_dagstore.go b/node/modules/storageminer_dagstore.go deleted file mode 100644 index 27575467d..000000000 --- a/node/modules/storageminer_dagstore.go +++ /dev/null @@ -1,138 +0,0 @@ -package modules - -import ( - "context" - "github.com/filecoin-project/boost-gfm/piecestore" - "github.com/filecoin-project/boost-gfm/storagemarket" - "github.com/filecoin-project/boost-gfm/stores" - "github.com/filecoin-project/boost/node/modules/dtypes" - "github.com/filecoin-project/dagstore" - lotus_gfm_piecestore "github.com/filecoin-project/go-fil-markets/piecestore" - "github.com/filecoin-project/go-fil-markets/shared" - lotus_gfm_storagemarket "github.com/filecoin-project/go-fil-markets/storagemarket" - mdagstore "github.com/filecoin-project/lotus/markets/dagstore" - lotus_dtypes "github.com/filecoin-project/lotus/node/modules/dtypes" - "github.com/ipfs/go-cid" - "github.com/ipld/go-car/v2/index" -) - -func NewBoostGFMDAGStoreWrapper(w *mdagstore.Wrapper) stores.DAGStoreWrapper { - return &boostDagstoreWrapper{w: w} -} - -type boostDagstoreWrapper struct { - w *mdagstore.Wrapper -} - -func (b *boostDagstoreWrapper) RegisterShard(ctx context.Context, pieceCid cid.Cid, carPath string, eagerInit bool, resch chan dagstore.ShardResult) error { - return b.w.RegisterShard(ctx, pieceCid, carPath, eagerInit, resch) -} - -func (b *boostDagstoreWrapper) LoadShard(ctx context.Context, pieceCid cid.Cid) (stores.ClosableBlockstore, error) { - return b.w.LoadShard(ctx, pieceCid) -} - -func (b *boostDagstoreWrapper) MigrateDeals(ctx context.Context, deals []storagemarket.MinerDeal) (bool, error) { - dls := make([]lotus_gfm_storagemarket.MinerDeal, 0, len(deals)) - for _, d := range deals { - dls = append(dls, toLotusGFMMinerDeal(d)) - } - return b.w.MigrateDeals(ctx, dls) -} - -func (b *boostDagstoreWrapper) GetPiecesContainingBlock(blockCID cid.Cid) ([]cid.Cid, error) { - return b.w.GetPiecesContainingBlock(blockCID) -} - -func (b *boostDagstoreWrapper) GetIterableIndexForPiece(pieceCid cid.Cid) (index.IterableIndex, error) { - return b.w.GetIterableIndexForPiece(pieceCid) -} - -func (b *boostDagstoreWrapper) DestroyShard(ctx context.Context, pieceCid cid.Cid, resch chan dagstore.ShardResult) error { - return b.w.DestroyShard(ctx, pieceCid, resch) -} - -func (b *boostDagstoreWrapper) Close() error { - return b.w.Close() -} - -func NewLotusGFMProviderPieceStore(ps dtypes.ProviderPieceStore) lotus_dtypes.ProviderPieceStore { - return &lotusProviderPieceStore{ProviderPieceStore: ps} -} - -type lotusProviderPieceStore struct { - dtypes.ProviderPieceStore -} - -var _ lotus_dtypes.ProviderPieceStore = (*lotusProviderPieceStore)(nil) - -func (l *lotusProviderPieceStore) OnReady(ready shared.ReadyFunc) { - if ready == nil { - return - } - l.ProviderPieceStore.OnReady(func(err error) { - ready(err) - }) -} - -func (l *lotusProviderPieceStore) AddDealForPiece(pieceCID cid.Cid, payloadCid cid.Cid, dealInfo lotus_gfm_piecestore.DealInfo) error { - return l.ProviderPieceStore.AddDealForPiece(pieceCID, payloadCid, piecestore.DealInfo{ - DealID: dealInfo.DealID, - SectorID: dealInfo.SectorID, - Offset: dealInfo.Offset, - Length: dealInfo.Length, - }) -} - -func (l *lotusProviderPieceStore) AddPieceBlockLocations(pieceCID cid.Cid, blockLocations map[cid.Cid]lotus_gfm_piecestore.BlockLocation) error { - bls := make(map[cid.Cid]piecestore.BlockLocation, len(blockLocations)) - for c, bl := range blockLocations { - bls[c] = piecestore.BlockLocation{ - RelOffset: bl.RelOffset, - BlockSize: bl.BlockSize, - } - } - return l.ProviderPieceStore.AddPieceBlockLocations(pieceCID, bls) -} - -func (l *lotusProviderPieceStore) GetPieceInfo(pieceCID cid.Cid) (lotus_gfm_piecestore.PieceInfo, error) { - pi, err := l.ProviderPieceStore.GetPieceInfo(pieceCID) - if err != nil { - return lotus_gfm_piecestore.PieceInfo{}, err - } - dls := make([]lotus_gfm_piecestore.DealInfo, 0, len(pi.Deals)) - for _, d := range pi.Deals { - dls = append(dls, lotus_gfm_piecestore.DealInfo{ - DealID: d.DealID, - SectorID: d.SectorID, - Offset: d.Offset, - Length: d.Length, - }) - } - return lotus_gfm_piecestore.PieceInfo{ - PieceCID: pi.PieceCID, - Deals: dls, - }, nil -} - -func (l *lotusProviderPieceStore) GetCIDInfo(payloadCID cid.Cid) (lotus_gfm_piecestore.CIDInfo, error) { - ci, err := l.ProviderPieceStore.GetCIDInfo(payloadCID) - if err != nil { - return lotus_gfm_piecestore.CIDInfo{}, err - } - - bls := make([]lotus_gfm_piecestore.PieceBlockLocation, 0, len(ci.PieceBlockLocations)) - for _, bl := range ci.PieceBlockLocations { - bls = append(bls, lotus_gfm_piecestore.PieceBlockLocation{ - BlockLocation: lotus_gfm_piecestore.BlockLocation{ - RelOffset: bl.BlockLocation.RelOffset, - BlockSize: bl.BlockLocation.BlockSize, - }, - PieceCID: bl.PieceCID, - }) - } - return lotus_gfm_piecestore.CIDInfo{ - CID: ci.CID, - PieceBlockLocations: bls, - }, nil -} diff --git a/node/modules/storageminer_idxprov.go b/node/modules/storageminer_idxprov.go index ac4c40751..401bdcb99 100644 --- a/node/modules/storageminer_idxprov.go +++ b/node/modules/storageminer_idxprov.go @@ -3,28 +3,20 @@ package modules import ( "context" "fmt" + "github.com/filecoin-project/boost/build" "github.com/filecoin-project/boost/indexprovider" "github.com/filecoin-project/boost/node/config" "github.com/filecoin-project/boost/node/modules/dtypes" - "github.com/filecoin-project/boost/retrievalmarket/types" "github.com/filecoin-project/boost/util" "github.com/filecoin-project/go-address" - datatransfer "github.com/filecoin-project/go-data-transfer" - "github.com/filecoin-project/go-data-transfer/transport/graphsync" - datatransferv2 "github.com/filecoin-project/go-data-transfer/v2" lotus_dtypes "github.com/filecoin-project/lotus/node/modules/dtypes" - "github.com/ipfs/go-cid" "github.com/ipfs/go-datastore" "github.com/ipfs/go-datastore/namespace" - "github.com/ipld/go-ipld-prime" - "github.com/ipld/go-ipld-prime/datamodel" - "github.com/ipni/go-libipni/dagsync/dtsync" provider "github.com/ipni/index-provider" "github.com/ipni/index-provider/engine" pubsub "github.com/libp2p/go-libp2p-pubsub" "github.com/libp2p/go-libp2p/core/host" - "github.com/libp2p/go-libp2p/core/peer" "go.uber.org/fx" "golang.org/x/xerrors" ) @@ -36,14 +28,14 @@ type IdxProv struct { Datastore lotus_dtypes.MetadataDS } -func IndexProvider(cfg config.IndexProviderConfig) func(params IdxProv, marketHost host.Host, dt dtypes.ProviderDataTransfer, maddr lotus_dtypes.MinerAddress, ps *pubsub.PubSub, nn lotus_dtypes.NetworkName) (provider.Interface, error) { +func IndexProvider(cfg config.IndexProviderConfig) func(params IdxProv, marketHost host.Host, maddr lotus_dtypes.MinerAddress, ps *pubsub.PubSub, nn lotus_dtypes.NetworkName) (provider.Interface, error) { if !cfg.Enable { log.Warnf("Starting Boost with index provider disabled - no announcements will be made to the index provider") - return func(params IdxProv, marketHost host.Host, dt dtypes.ProviderDataTransfer, maddr lotus_dtypes.MinerAddress, ps *pubsub.PubSub, nn lotus_dtypes.NetworkName) (provider.Interface, error) { + return func(params IdxProv, marketHost host.Host, maddr lotus_dtypes.MinerAddress, ps *pubsub.PubSub, nn lotus_dtypes.NetworkName) (provider.Interface, error) { return indexprovider.NewDisabledIndexProvider(), nil } } - return func(args IdxProv, marketHost host.Host, dt dtypes.ProviderDataTransfer, maddr lotus_dtypes.MinerAddress, ps *pubsub.PubSub, nn lotus_dtypes.NetworkName) (provider.Interface, error) { + return func(args IdxProv, marketHost host.Host, maddr lotus_dtypes.MinerAddress, ps *pubsub.PubSub, nn lotus_dtypes.NetworkName) (provider.Interface, error) { topicName := cfg.TopicName // If indexer topic name is left empty, infer it from the network name. if topicName == "" { @@ -102,17 +94,8 @@ func IndexProvider(cfg config.IndexProviderConfig) func(params IdxProv, marketHo opts = append(opts, engine.WithDirectAnnounce(cfg.Announce.DirectAnnounceURLs...)) } - // Advertisements can be served over HTTP, HTTP over libp2p of over - // the data transfer protocol (on graphsync). - if cfg.DataTransferPublisher { - opts = append(opts, - engine.WithPublisherKind(engine.DataTransferPublisher), - engine.WithDataTransfer(dtV1ToIndexerDT(dt, func() ipld.LinkSystem { - return *e.LinkSystem() - })), - ) - llog = llog.With("extraGossipData", ma, "publisher", "data-transfer") - } else if cfg.HttpPublisher.Enabled { + // Advertisements can be served over HTTP or HTTP over libp2p. + if cfg.HttpPublisher.Enabled { announceAddr, err := util.ToHttpMultiaddr(cfg.HttpPublisher.PublicHostname, cfg.HttpPublisher.Port) if err != nil { return nil, fmt.Errorf("parsing HTTP Publisher hostname '%s' / port %d: %w", @@ -130,7 +113,7 @@ func IndexProvider(cfg config.IndexProviderConfig) func(params IdxProv, marketHo llog = llog.With("publisher", "http and libp2phttp", "announceAddr", announceAddr, "extraGossipData", ma) } } else { - // HTTP publisher not enabled, so use only libp2p unless using DataTransferPublisher. + // HTTP publisher not enabled, so use only libp2p opts = append(opts, engine.WithPublisherKind(engine.Libp2pPublisher)) llog = llog.With("publisher", "libp2phttp", "extraGossipData", ma) } @@ -168,156 +151,3 @@ func IndexProvider(cfg config.IndexProviderConfig) func(params IdxProv, marketHo return e, nil } } - -// The index provider needs to set up some go-data-transfer voucher code. -// Below we write a shim for the specific use case of index provider, that -// translates between the go-data-transfer v2 use case that the index provider -// implements and the go-data-transfer v1 code that boost imports. -func dtV1ToIndexerDT(dt dtypes.ProviderDataTransfer, linksys func() ipld.LinkSystem) datatransferv2.Manager { - return &indexerDT{dt: dt, linksys: linksys} -} - -type indexerDT struct { - dt dtypes.ProviderDataTransfer - linksys func() ipld.LinkSystem -} - -var _ datatransferv2.Manager = (*indexerDT)(nil) - -func (i *indexerDT) RegisterVoucherType(voucherType datatransferv2.TypeIdentifier, validator datatransferv2.RequestValidator) error { - if voucherType == dtsync.LegsVoucherType { - return i.dt.RegisterVoucherType(&types.LegsVoucherDTv1{}, &dtv1ReqValidator{v: validator}) - } - return fmt.Errorf("unrecognized voucher type: %s", voucherType) -} - -func (i *indexerDT) RegisterTransportConfigurer(voucherType datatransferv2.TypeIdentifier, configurer datatransferv2.TransportConfigurer) error { - if voucherType == dtsync.LegsVoucherType { - return i.dt.RegisterTransportConfigurer(&types.LegsVoucherDTv1{}, func(chid datatransfer.ChannelID, voucher datatransfer.Voucher, transport datatransfer.Transport) { - gsTransport, ok := transport.(*graphsync.Transport) - if ok { - err := gsTransport.UseStore(chid, i.linksys()) - if err != nil { - log.Warnf("setting store for legs voucher: %s", err) - } - } else { - log.Warnf("expected transport configurer to pass graphsync transport but got %T", transport) - } - }) - } - return fmt.Errorf("unrecognized voucher type: %s", voucherType) -} - -func (i *indexerDT) Start(ctx context.Context) error { - return fmt.Errorf("not implemented") -} - -func (i *indexerDT) OnReady(readyFunc datatransferv2.ReadyFunc) { -} - -func (i *indexerDT) Stop(ctx context.Context) error { - return fmt.Errorf("not implemented") -} - -func (i *indexerDT) OpenPushDataChannel(ctx context.Context, to peer.ID, voucher datatransferv2.TypedVoucher, baseCid cid.Cid, selector datamodel.Node, options ...datatransferv2.TransferOption) (datatransferv2.ChannelID, error) { - return datatransferv2.ChannelID{}, fmt.Errorf("not implemented") -} - -func (i *indexerDT) OpenPullDataChannel(ctx context.Context, to peer.ID, voucher datatransferv2.TypedVoucher, baseCid cid.Cid, selector datamodel.Node, options ...datatransferv2.TransferOption) (datatransferv2.ChannelID, error) { - return datatransferv2.ChannelID{}, fmt.Errorf("not implemented") -} - -func (i *indexerDT) SendVoucher(ctx context.Context, chid datatransferv2.ChannelID, voucher datatransferv2.TypedVoucher) error { - return fmt.Errorf("not implemented") -} - -func (i *indexerDT) SendVoucherResult(ctx context.Context, chid datatransferv2.ChannelID, voucherResult datatransferv2.TypedVoucher) error { - return fmt.Errorf("not implemented") -} - -func (i *indexerDT) UpdateValidationStatus(ctx context.Context, chid datatransferv2.ChannelID, validationResult datatransferv2.ValidationResult) error { - return fmt.Errorf("not implemented") -} - -func (i *indexerDT) CloseDataTransferChannel(ctx context.Context, chid datatransferv2.ChannelID) error { - return fmt.Errorf("not implemented") -} - -func (i *indexerDT) PauseDataTransferChannel(ctx context.Context, chid datatransferv2.ChannelID) error { - return fmt.Errorf("not implemented") -} - -func (i *indexerDT) ResumeDataTransferChannel(ctx context.Context, chid datatransferv2.ChannelID) error { - return fmt.Errorf("not implemented") -} - -func (i *indexerDT) TransferChannelStatus(ctx context.Context, x datatransferv2.ChannelID) datatransferv2.Status { - return 0 -} - -func (i *indexerDT) ChannelState(ctx context.Context, chid datatransferv2.ChannelID) (datatransferv2.ChannelState, error) { - return nil, fmt.Errorf("not implemented") -} - -func (i *indexerDT) SubscribeToEvents(subscriber datatransferv2.Subscriber) datatransferv2.Unsubscribe { - return func() {} -} - -func (i *indexerDT) InProgressChannels(ctx context.Context) (map[datatransferv2.ChannelID]datatransferv2.ChannelState, error) { - return nil, fmt.Errorf("not implemented") -} - -func (i *indexerDT) RestartDataTransferChannel(ctx context.Context, chid datatransferv2.ChannelID) error { - return fmt.Errorf("not implemented") -} - -type dtv1ReqValidator struct { - v datatransferv2.RequestValidator -} - -func (d *dtv1ReqValidator) ValidatePush(isRestart bool, chid datatransfer.ChannelID, sender peer.ID, voucher datatransfer.Voucher, baseCid cid.Cid, selector ipld.Node) (datatransfer.VoucherResult, error) { - d2v := dtsync.BindnodeRegistry.TypeToNode(&voucher.(*types.LegsVoucherDTv1).Voucher) - res, err := d.v.ValidatePush(toChannelIDV2(chid), sender, d2v, baseCid, selector) - if err != nil { - return nil, err - } - if !res.Accepted { - return nil, datatransfer.ErrRejected - } - - return toVoucherResult(res) -} - -func (d *dtv1ReqValidator) ValidatePull(isRestart bool, chid datatransfer.ChannelID, receiver peer.ID, voucher datatransfer.Voucher, baseCid cid.Cid, selector ipld.Node) (datatransfer.VoucherResult, error) { - d2v := dtsync.BindnodeRegistry.TypeToNode(&voucher.(*types.LegsVoucherDTv1).Voucher) - res, err := d.v.ValidatePull(toChannelIDV2(chid), receiver, d2v, baseCid, selector) - if err != nil { - return nil, err - } - if !res.Accepted { - return nil, datatransfer.ErrRejected - } - - return toVoucherResult(res) -} - -func toVoucherResult(res datatransferv2.ValidationResult) (datatransfer.VoucherResult, error) { - voucherResVoucher := res.VoucherResult.Voucher - vri, err := dtsync.BindnodeRegistry.TypeFromNode(voucherResVoucher, &dtsync.VoucherResult{}) - if err != nil { - return nil, fmt.Errorf("getting VoucherResult from ValidationResult: %w", err) - } - vr := vri.(*dtsync.VoucherResult) - if vr == nil { - return nil, fmt.Errorf("got nil VoucherResult from ValidationResult") - } - return &types.LegsVoucherResultDtv1{VoucherResult: *vr, VoucherType: res.VoucherResult.Type}, nil -} - -func toChannelIDV2(chid datatransfer.ChannelID) datatransferv2.ChannelID { - return datatransferv2.ChannelID{ - Initiator: chid.Initiator, - Responder: chid.Responder, - ID: datatransferv2.TransferID(chid.ID), - } -} diff --git a/react/src/StorageSpace.js b/react/src/StorageSpace.js index de13fa98e..e8ad7ec31 100644 --- a/react/src/StorageSpace.js +++ b/react/src/StorageSpace.js @@ -1,5 +1,5 @@ import {useQuery} from "@apollo/react-hooks"; -import {LegacyStorageQuery, StorageQuery} from "./gql"; +import {StorageQuery} from "./gql"; import React from "react"; import {addCommas, humanFileSize} from "./util"; import './StorageSpace.css' @@ -12,7 +12,6 @@ import {Info} from "./Info" export function StorageSpacePage(props) { return - } @@ -81,61 +80,6 @@ function StorageSpaceContent(props) { } -function LegacyStorageSpaceContent(props) { - const {loading, error, data} = useQuery(LegacyStorageQuery, { pollInterval: 10000 }) - - if (loading) { - return
Loading...
- } - if (error) { - return
Error: {error.message}
- } - - var storage = data.legacyStorage - if (storage.Capacity === 0n) { - return null - } - - const bars = [{ - name: 'Used', - className: 'used', - amount: storage.Used, - }, { - name: 'Free', - className: 'free', - amount: storage.Capacity - storage.Used, - }] - - return
-

Legacy Deal transfers

- -
- - -
- - - - {bars.map(bar => ( - - - - - ))} - - - - - -
- {bar.name} - {humanFileSize(bar.amount)} ({addCommas(bar.amount)} bytes)
- Mount Point - The path to the directory where downloaded data is kept until the deal is added to a sector - {storage.MountPoint}
-
-} - export function StorageSpaceMenuItem(props) { const {data} = useQuery(StorageQuery, { pollInterval: 10000, diff --git a/react/src/gql.js b/react/src/gql.js index 4d5c3b879..62361d78a 100644 --- a/react/src/gql.js +++ b/react/src/gql.js @@ -646,16 +646,6 @@ const StorageQuery = gql` } `; -const LegacyStorageQuery = gql` - query AppLegacyStorageQuery { - legacyStorage { - Capacity - Used - MountPoint - } - } -`; - const SealingPipelineQuery = gql` query AppSealingPipelineQuery { sealingpipeline { @@ -924,7 +914,6 @@ export { FlaggedPiecesCountQuery, LIDQuery, StorageQuery, - LegacyStorageQuery, FundsQuery, FundsLogsQuery, DealPublishQuery, diff --git a/retrievalmarket/client/client.go b/retrievalmarket/client/client.go index 982c28414..23eac8e82 100644 --- a/retrievalmarket/client/client.go +++ b/retrievalmarket/client/client.go @@ -9,27 +9,29 @@ import ( "sync" "time" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/boost-gfm/shared" gsimpl "github.com/filecoin-project/boost-graphsync/impl" gsnet "github.com/filecoin-project/boost-graphsync/network" "github.com/filecoin-project/boost-graphsync/storeutil" + datatransfer2 "github.com/filecoin-project/boost/datatransfer" + dtnet "github.com/filecoin-project/boost/datatransfer/network" + gst "github.com/filecoin-project/boost/datatransfer/transport/graphsync" + "github.com/filecoin-project/boost/markets/shared" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" "github.com/filecoin-project/go-address" cborutil "github.com/filecoin-project/go-cbor-util" - datatransfer "github.com/filecoin-project/go-data-transfer" - "github.com/filecoin-project/go-data-transfer/channelmonitor" - dtimpl "github.com/filecoin-project/go-data-transfer/impl" - dtnet "github.com/filecoin-project/go-data-transfer/network" - gst "github.com/filecoin-project/go-data-transfer/transport/graphsync" + + "github.com/filecoin-project/boost/datatransfer/channelmonitor" + dtimpl "github.com/filecoin-project/boost/datatransfer/impl" "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/lotus/api" "github.com/filecoin-project/lotus/chain/types" "github.com/filecoin-project/lotus/chain/wallet" - blockstore "github.com/ipfs/boxo/blockstore" + "github.com/ipfs/boxo/blockstore" "github.com/ipfs/go-cid" "github.com/ipfs/go-datastore" logging "github.com/ipfs/go-log/v2" "github.com/ipld/go-ipld-prime" + "github.com/ipld/go-ipld-prime/codec/dagcbor" selectorparse "github.com/ipld/go-ipld-prime/traversal/selector/parse" "github.com/libp2p/go-libp2p/core/host" inet "github.com/libp2p/go-libp2p/core/network" @@ -61,7 +63,7 @@ type Client struct { host host.Host ClientAddr address.Address blockstore blockstore.Blockstore - dataTransfer datatransfer.Manager + dataTransfer datatransfer2.Manager logRetrievalProgressEvents bool } @@ -76,7 +78,7 @@ type Config struct { Datastore datastore.Batching Host host.Host ChannelMonitorConfig channelmonitor.Config - RetrievalConfigurer datatransfer.TransportConfigurer + RetrievalConfigurer datatransfer2.TransportConfigurer LogRetrievalProgressEvents bool } @@ -138,31 +140,48 @@ func NewClientWithConfig(cfg *Config) (*Client, error) { return nil, err } - err = mgr.RegisterVoucherType(&retrievalmarket.DealProposal{}, nil) + err = mgr.RegisterVoucherType(&legacyretrievaltypes.DealProposal{}, nil) if err != nil { return nil, err } - err = mgr.RegisterVoucherType(&retrievalmarket.DealPayment{}, nil) + err = mgr.RegisterVoucherType(&legacyretrievaltypes.DealPayment{}, nil) if err != nil { return nil, err } - err = mgr.RegisterVoucherResultType(&retrievalmarket.DealResponse{}) + err = mgr.RegisterVoucherResultType(&legacyretrievaltypes.DealResponse{}) if err != nil { return nil, err } if cfg.RetrievalConfigurer != nil { - if err := mgr.RegisterTransportConfigurer(&retrievalmarket.DealProposal{}, cfg.RetrievalConfigurer); err != nil { + if err := mgr.RegisterTransportConfigurer(&legacyretrievaltypes.DealProposal{}, cfg.RetrievalConfigurer); err != nil { return nil, err } } + errCh := make(chan error) + startedCh := make(chan struct{}) + + mgr.OnReady(func(err error) { + if err != nil { + errCh <- err + return + } + close(startedCh) + }) + if err := mgr.Start(context.Background()); err != nil { return nil, err } + select { + case <-startedCh: + case err := <-errCh: + return nil, err + } + c := &Client{ host: cfg.Host, api: cfg.Api, @@ -297,7 +316,7 @@ func doRpc(ctx context.Context, s inet.Stream, req interface{}, resp interface{} return nil } -func (c *Client) RetrievalQuery(ctx context.Context, maddr address.Address, pcid cid.Cid) (*retrievalmarket.QueryResponse, error) { +func (c *Client) RetrievalQuery(ctx context.Context, maddr address.Address, pcid cid.Cid) (*legacyretrievaltypes.QueryResponse, error) { ctx, span := Tracer.Start(ctx, "retrievalQuery", trace.WithAttributes( attribute.Stringer("miner", maddr), )) @@ -316,11 +335,11 @@ func (c *Client) RetrievalQuery(ctx context.Context, maddr address.Address, pcid // We have connected - q := &retrievalmarket.Query{ + q := &legacyretrievaltypes.Query{ PayloadCID: pcid, } - var resp retrievalmarket.QueryResponse + var resp legacyretrievaltypes.QueryResponse if err := doRpc(ctx, s, q, &resp); err != nil { return nil, fmt.Errorf("retrieval query rpc: %w", err) } @@ -341,7 +360,7 @@ type RetrievalStats struct { func (c *Client) RetrieveContentWithProgressCallback( ctx context.Context, miner address.Address, - proposal *retrievalmarket.DealProposal, + proposal *legacyretrievaltypes.DealProposal, progressCallback func(bytesReceived uint64), ) (*RetrievalStats, error) { @@ -362,7 +381,7 @@ func (c *Client) retrieveContentFromPeerWithProgressCallback( ctx context.Context, peerID peer.ID, minerWallet address.Address, - proposal *retrievalmarket.DealProposal, + proposal *legacyretrievaltypes.DealProposal, progressCallback func(bytesReceived uint64), gracefulShutdownRequested <-chan struct{}, ) (*RetrievalStats, error) { @@ -380,7 +399,7 @@ func (c *Client) retrieveContentFromPeerWithProgressCallback( totalPayment := abi.NewTokenAmount(0) rootCid := proposal.PayloadCID - var chanid datatransfer.ChannelID + var chanid datatransfer2.ChannelID var chanidLk sync.Mutex pchRequired := !proposal.PricePerByte.IsZero() || !proposal.UnsealPrice.IsZero() @@ -409,8 +428,8 @@ func (c *Client) retrieveContentFromPeerWithProgressCallback( dealComplete := false receivedFirstByte := false - unsubscribe := c.dataTransfer.SubscribeToEvents(func(event datatransfer.Event, state datatransfer.ChannelState) { - // Copy chanid so it can be used later in the callback + unsubscribe := c.dataTransfer.SubscribeToEvents(func(event datatransfer2.Event, state datatransfer2.ChannelState) { + // Copy chanid so, it can be used later in the callback chanidLk.Lock() chanidCopy := chanid chanidLk.Unlock() @@ -424,24 +443,24 @@ func (c *Client) retrieveContentFromPeerWithProgressCallback( eventCodeNotHandled := false switch event.Code { - case datatransfer.Open: - case datatransfer.Accept: - case datatransfer.Restart: - case datatransfer.DataReceived: + case datatransfer2.Open: + case datatransfer2.Accept: + case datatransfer2.Restart: + case datatransfer2.DataReceived: silenceEventCode = true - case datatransfer.DataSent: - case datatransfer.Cancel: - case datatransfer.Error: + case datatransfer2.DataSent: + case datatransfer2.Cancel: + case datatransfer2.Error: finish(fmt.Errorf("datatransfer error: %s", event.Message)) return - case datatransfer.CleanupComplete: + case datatransfer2.CleanupComplete: finish(nil) return - case datatransfer.NewVoucher: - case datatransfer.NewVoucherResult: + case datatransfer2.NewVoucher: + case datatransfer2.NewVoucherResult: switch resType := state.LastVoucherResult().(type) { - case *retrievalmarket.DealResponse: + case *legacyretrievaltypes.DealResponse: if len(resType.Message) != 0 { log.Debugf("Received deal response voucher result %s (%v): %s\n\t%+v", resType.Status, resType.Status, resType.Message, resType) } else { @@ -449,11 +468,11 @@ func (c *Client) retrieveContentFromPeerWithProgressCallback( } switch resType.Status { - case retrievalmarket.DealStatusAccepted: + case legacyretrievaltypes.DealStatusAccepted: log.Info("Deal accepted") // Respond with a payment voucher when funds are requested - case retrievalmarket.DealStatusFundsNeeded, retrievalmarket.DealStatusFundsNeededLastPayment: + case legacyretrievaltypes.DealStatusFundsNeeded, legacyretrievaltypes.DealStatusFundsNeededLastPayment: if pchRequired { finish(errors.New("payment channel required")) return @@ -461,19 +480,19 @@ func (c *Client) retrieveContentFromPeerWithProgressCallback( finish(fmt.Errorf("the miner requested payment even though this transaction was determined to be zero cost")) return } - case retrievalmarket.DealStatusRejected: + case legacyretrievaltypes.DealStatusRejected: finish(fmt.Errorf("deal rejected: %s", resType.Message)) return - case retrievalmarket.DealStatusFundsNeededUnseal, retrievalmarket.DealStatusUnsealing: + case legacyretrievaltypes.DealStatusFundsNeededUnseal, legacyretrievaltypes.DealStatusUnsealing: finish(fmt.Errorf("data is sealed")) return - case retrievalmarket.DealStatusCancelled: + case legacyretrievaltypes.DealStatusCancelled: finish(fmt.Errorf("deal cancelled: %s", resType.Message)) return - case retrievalmarket.DealStatusErrored: + case legacyretrievaltypes.DealStatusErrored: finish(fmt.Errorf("deal errored: %s", resType.Message)) return - case retrievalmarket.DealStatusCompleted: + case legacyretrievaltypes.DealStatusCompleted: if allBytesReceived { finish(nil) return @@ -481,26 +500,26 @@ func (c *Client) retrieveContentFromPeerWithProgressCallback( dealComplete = true } } - case datatransfer.PauseInitiator: - case datatransfer.ResumeInitiator: - case datatransfer.PauseResponder: - case datatransfer.ResumeResponder: - case datatransfer.FinishTransfer: + case datatransfer2.PauseInitiator: + case datatransfer2.ResumeInitiator: + case datatransfer2.PauseResponder: + case datatransfer2.ResumeResponder: + case datatransfer2.FinishTransfer: if dealComplete { finish(nil) return } allBytesReceived = true - case datatransfer.ResponderCompletes: - case datatransfer.ResponderBeginsFinalization: - case datatransfer.BeginFinalizing: - case datatransfer.Disconnected: - case datatransfer.Complete: - case datatransfer.CompleteCleanupOnRestart: - case datatransfer.DataQueued: - case datatransfer.DataQueuedProgress: - case datatransfer.DataSentProgress: - case datatransfer.DataReceivedProgress: + case datatransfer2.ResponderCompletes: + case datatransfer2.ResponderBeginsFinalization: + case datatransfer2.BeginFinalizing: + case datatransfer2.Disconnected: + case datatransfer2.Complete: + case datatransfer2.CompleteCleanupOnRestart: + case datatransfer2.DataQueued: + case datatransfer2.DataQueuedProgress: + case datatransfer2.DataSentProgress: + case datatransfer2.DataReceivedProgress: // First byte has been received // publish first byte event @@ -510,17 +529,17 @@ func (c *Client) retrieveContentFromPeerWithProgressCallback( progressCallback(state.Received()) silenceEventCode = true - case datatransfer.RequestTimedOut: - case datatransfer.SendDataError: - case datatransfer.ReceiveDataError: - case datatransfer.TransferRequestQueued: - case datatransfer.RequestCancelled: - case datatransfer.Opened: + case datatransfer2.RequestTimedOut: + case datatransfer2.SendDataError: + case datatransfer2.ReceiveDataError: + case datatransfer2.TransferRequestQueued: + case datatransfer2.RequestCancelled: + case datatransfer2.Opened: default: eventCodeNotHandled = true } - name := datatransfer.Events[event.Code] + name := datatransfer2.Events[event.Code] code := event.Code msg := event.Message blocksIndex := state.ReceivedCidsTotal() @@ -536,7 +555,15 @@ func (c *Client) retrieveContentFromPeerWithProgressCallback( defer unsubscribe() // Submit the retrieval deal proposal to the miner - newchid, err := c.dataTransfer.OpenPullDataChannel(ctx, peerID, proposal, proposal.PayloadCID, selectorparse.CommonSelector_ExploreAllRecursively) + selector := selectorparse.CommonSelector_ExploreAllRecursively + if proposal.SelectorSpecified() { + var err error + selector, err = ipld.Decode(proposal.Selector.Raw, dagcbor.Decode) + if err != nil { + return nil, fmt.Errorf("failed to decode selector from proposal: %w", err) + } + } + newchid, err := c.dataTransfer.OpenPullDataChannel(ctx, peerID, proposal, proposal.PayloadCID, selector) if err != nil { // We could fail before a successful proposal // publish event failure @@ -602,12 +629,12 @@ awaitfinished: }, nil } -func RetrievalProposalForAsk(ask *retrievalmarket.QueryResponse, c cid.Cid, optionalSelector ipld.Node) (*retrievalmarket.DealProposal, error) { +func RetrievalProposalForAsk(ask *legacyretrievaltypes.QueryResponse, c cid.Cid, optionalSelector ipld.Node) (*legacyretrievaltypes.DealProposal, error) { if optionalSelector == nil { optionalSelector = selectorparse.CommonSelector_ExploreAllRecursively } - params, err := retrievalmarket.NewParamsV1( + params, err := legacyretrievaltypes.NewParamsV1( ask.MinPricePerByte, ask.MaxPaymentInterval, ask.MaxPaymentIntervalIncrease, @@ -618,9 +645,9 @@ func RetrievalProposalForAsk(ask *retrievalmarket.QueryResponse, c cid.Cid, opti if err != nil { return nil, err } - return &retrievalmarket.DealProposal{ + return &legacyretrievaltypes.DealProposal{ PayloadCID: c, - ID: retrievalmarket.DealID(dealIdGen.Next()), + ID: legacyretrievaltypes.DealID(dealIdGen.Next()), Params: params, }, nil } diff --git a/retrievalmarket/lib/idxciddagstore.go b/retrievalmarket/lib/idxciddagstore.go deleted file mode 100644 index 6f651d1f4..000000000 --- a/retrievalmarket/lib/idxciddagstore.go +++ /dev/null @@ -1,72 +0,0 @@ -package lib - -import ( - "context" - "fmt" - "github.com/filecoin-project/boost/retrievalmarket/server" - "github.com/filecoin-project/dagstore" - "github.com/filecoin-project/dagstore/indexbs" - "github.com/filecoin-project/dagstore/shard" - "github.com/ipfs/go-cid" - "github.com/multiformats/go-multihash" -) - -// IndexBackedBlockstoreDagstore implements the dagstore interface needed -// by the IndexBackedBlockstore. -// The implementation of ShardsContainingCid handles identity cids. -type IndexBackedBlockstoreDagstore struct { - dagstore.Interface -} - -var _ dagstore.Interface = (*IndexBackedBlockstoreDagstore)(nil) - -func NewIndexBackedBlockstoreDagstore(ds dagstore.Interface) indexbs.IdxBstoreDagstore { - return &IndexBackedBlockstoreDagstore{Interface: ds} -} - -// ShardsContainingCid checks the db for shards containing the given cid. -// If there are no shards with that cid, it checks if the shard is an identity -// cid, and gets the shards containing the identity cid's child cids. -// This is for the case where the identity cid was not stored in the original -// CAR file's index (but the identity cid's child cids are in the index). -func (i *IndexBackedBlockstoreDagstore) ShardsContainingCid(ctx context.Context, c cid.Cid) ([]shard.Key, error) { - shards, err := i.Interface.ShardsContainingMultihash(ctx, c.Hash()) - if err == nil { - return shards, nil - } - - var idErr error - piecesWithTargetBlock, idErr := server.GetCommonPiecesFromIdentityCidLinks(ctx, func(ctx context.Context, mh multihash.Multihash) ([]cid.Cid, error) { - return i.piecesContainingBlock(ctx, mh) - }, c) - if idErr != nil { - return nil, fmt.Errorf("getting common pieces for cid %s: %w", c, idErr) - } - if len(piecesWithTargetBlock) == 0 { - // No pieces found for cid: return the original error from the call to - // ShardsContainingMultihash above - return nil, fmt.Errorf("getting pieces for cid %s: %w", c, err) - } - - shards = make([]shard.Key, 0, len(piecesWithTargetBlock)) - for _, pcid := range piecesWithTargetBlock { - shards = append(shards, shard.KeyFromCID(pcid)) - } - return shards, nil -} - -func (i *IndexBackedBlockstoreDagstore) piecesContainingBlock(ctx context.Context, mh multihash.Multihash) ([]cid.Cid, error) { - shards, err := i.Interface.ShardsContainingMultihash(ctx, mh) - if err != nil { - return nil, fmt.Errorf("finding shards containing child mh %s: %w", mh, err) - } - pcids := make([]cid.Cid, 0, len(shards)) - for _, s := range shards { - pcid, err := cid.Parse(s.String()) - if err != nil { - return nil, fmt.Errorf("parsing shard into cid: %w", err) - } - pcids = append(pcids, pcid) - } - return pcids, nil -} diff --git a/retrievalmarket/lib/shardselector.go b/retrievalmarket/lib/shardselector.go deleted file mode 100644 index 2fe71c529..000000000 --- a/retrievalmarket/lib/shardselector.go +++ /dev/null @@ -1,181 +0,0 @@ -package lib - -import ( - "context" - "fmt" - "sync" - "time" - - "github.com/filecoin-project/boost-gfm/piecestore" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/dagstore/indexbs" - "github.com/filecoin-project/dagstore/shard" - "github.com/filecoin-project/go-state-types/abi" - lru "github.com/hnlq715/golang-lru" - "github.com/ipfs/go-cid" - logging "github.com/ipfs/go-log/v2" -) - -var sslog = logging.Logger("shardselect") - -// ShardSelector is used by the dagstore's index-backed blockstore to select -// the best shard from which to retrieve a particular cid. -// It chooses the first shard that is unsealed and free (zero cost). -// It caches the results per-shard. -type ShardSelector struct { - ctx context.Context - ps piecestore.PieceStore - sa retrievalmarket.SectorAccessor - rp retrievalmarket.RetrievalProvider - - // The striped lock protects against multiple threads doing a lookup - // against the sealing subsystem / retrieval ask for the same shard - stripedLock [256]sync.Mutex - cache *lru.Cache -} - -func NewShardSelector(ctx context.Context, ps piecestore.PieceStore, sa retrievalmarket.SectorAccessor, rp retrievalmarket.RetrievalProvider) (*ShardSelector, error) { - cache, err := lru.New(2048) - if err != nil { - return nil, fmt.Errorf("creating shard selector cache: %w", err) - } - return &ShardSelector{ctx: ctx, ps: ps, sa: sa, rp: rp, cache: cache}, nil -} - -var selectorCacheDuration = 10 * time.Minute -var selectorCacheErrorDuration = time.Minute - -type shardSelectResult struct { - available bool - err error -} - -// ShardSelectorF chooses the first shard that is unsealed and free (zero cost) -func (s *ShardSelector) ShardSelectorF(c cid.Cid, shards []shard.Key) (shard.Key, error) { - // If no shards are selected, return ErrNoShardSelected - lastErr := indexbs.ErrNoShardSelected - - sslog.Debugw("shard selection", "shards", shards) - for _, sk := range shards { - lkidx := s.stripedLockIndex(sk) - s.stripedLock[lkidx].Lock() - available, err := s.isAvailable(sk) - s.stripedLock[lkidx].Unlock() - - if available { - // We found an available shard, return it - sslog.Debugw("shard selected", "shard", sk) - return sk, nil - } - if err != nil { - sslog.Debugw("shard error", "shard", sk, "err", err) - lastErr = err - } - } - - // None of the shards are available - sslog.Debugw("no shard selected", "shards", shards, "err", lastErr) - return shard.Key{}, lastErr -} - -func (s *ShardSelector) isAvailable(sk shard.Key) (bool, error) { - // Check if the shard key is in the cache - var res *shardSelectResult - resi, cached := s.cache.Get(sk) - if cached { - res = resi.(*shardSelectResult) - sslog.Debugw("shard cache hit", "shard", sk) - return res.available, res.err - } - sslog.Debugw("shard cache miss", "shard", sk) - - // Check if the shard is available - res = &shardSelectResult{} - res.available, res.err = s.checkIsAvailable(sk) - expireIn := selectorCacheDuration - if res.err != nil { - // If there's an error, cache for a short duration so that we - // don't wait too long to try again. - expireIn = selectorCacheErrorDuration - res.available = false - res.err = fmt.Errorf("running shard selection for shard %s: %w", sk, res.err) - sslog.Warnw("checking shard availability", "shard", sk, "err", res.err) - } - // Add the result to the cache - s.cache.AddEx(sk, res, expireIn) - - return res.available, res.err -} - -func (s *ShardSelector) checkIsAvailable(sk shard.Key) (bool, error) { - // Parse piece CID - pieceCid, err := cid.Parse(sk.String()) - if err != nil { - return false, fmt.Errorf("parsing shard key as cid: %w", err) - } - - // Read piece info from piece store - sslog.Debugw("getting piece info", "shard", sk) - pieceInfo, err := s.ps.GetPieceInfo(pieceCid) - if err != nil { - return false, fmt.Errorf("get piece info: %w", err) - } - - // Filter for deals that are unsealed - sslog.Debugw("filtering for unsealed deals", "shard", sk, "deals", len(pieceInfo.Deals)) - unsealedDeals := make([]piecestore.DealInfo, 0, len(pieceInfo.Deals)) - var lastErr error - for _, di := range pieceInfo.Deals { - isUnsealed, err := s.sa.IsUnsealed(s.ctx, di.SectorID, di.Offset.Unpadded(), di.Length.Unpadded()) - if err != nil { - sslog.Warnf("checking if sector is unsealed", "shard", sk, "sector", di.SectorID, sk, "err", err) - lastErr = err - continue - } - - if isUnsealed { - sslog.Debugw("sector is unsealed", "shard", sk, "sector", di.SectorID) - unsealedDeals = append(unsealedDeals, di) - } else { - sslog.Debugw("sector is sealed", "shard", sk, "sector", di.SectorID) - } - } - - if len(unsealedDeals) == 0 { - // It wasn't possible to find an unsealed sector - sslog.Debugw("no unsealed deals found", "shard", sk) - return false, lastErr - } - - // Check if the piece is available for free (zero-cost) retrieval - input := retrievalmarket.PricingInput{ - // Piece from which the payload will be retrieved - PieceCID: pieceInfo.PieceCID, - Unsealed: true, - } - - var dealsIds []abi.DealID - for _, d := range unsealedDeals { - dealsIds = append(dealsIds, d.DealID) - } - - sslog.Debugw("getting dynamic asking price for unsealed deals", "shard", sk, "deals", len(unsealedDeals)) - ask, err := s.rp.GetDynamicAsk(s.ctx, input, dealsIds) - if err != nil { - return false, fmt.Errorf("getting retrieval ask: %w", err) - } - - // The piece is available for free retrieval - if ask.PricePerByte.NilOrZero() { - sslog.Debugw("asking price for unsealed deals is zero", "shard", sk) - return true, nil - } - - sslog.Debugw("asking price-per-byte for unsealed deals is non-zero", "shard", sk, "price", ask.PricePerByte.String()) - return false, nil -} - -func (s *ShardSelector) stripedLockIndex(sk shard.Key) int { - skstr := sk.String() - return int(skstr[len(skstr)-1]) -} diff --git a/retrievalmarket/lib/shardselector_test.go b/retrievalmarket/lib/shardselector_test.go deleted file mode 100644 index 7cbf99927..000000000 --- a/retrievalmarket/lib/shardselector_test.go +++ /dev/null @@ -1,129 +0,0 @@ -package lib - -import ( - "context" - "github.com/filecoin-project/boost-gfm/piecestore" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/boost/retrievalmarket/mock" - "github.com/filecoin-project/boost/testutil" - "github.com/filecoin-project/dagstore/indexbs" - "github.com/filecoin-project/dagstore/shard" - "github.com/filecoin-project/go-state-types/abi" - "github.com/golang/mock/gomock" - "github.com/stretchr/testify/require" - "golang.org/x/sync/errgroup" - "testing" -) - -func TestShardSelector(t *testing.T) { - ctx := context.Background() - - testCases := []struct { - name string - deals []piecestore.DealInfo - isUnsealed []bool - pricePerByte int64 - expectErr error - }{{ - name: "no deals", - deals: nil, - expectErr: indexbs.ErrNoShardSelected, - }, { - name: "only sealed deals", - deals: []piecestore.DealInfo{{SectorID: 0}, {SectorID: 1}}, - isUnsealed: []bool{false, false}, // index corresponds to sector ID - expectErr: indexbs.ErrNoShardSelected, - }, { - name: "one unsealed deal but non-zero price", - deals: []piecestore.DealInfo{{SectorID: 0}, {SectorID: 1}}, - isUnsealed: []bool{false, true}, // index corresponds to sector ID - pricePerByte: 1, - expectErr: indexbs.ErrNoShardSelected, - }, { - name: "one unsealed deal with zero price", - deals: []piecestore.DealInfo{{SectorID: 0}, {SectorID: 1}}, - isUnsealed: []bool{false, true}, // index corresponds to sector ID - pricePerByte: 0, - expectErr: nil, - }} - - for _, tc := range testCases { - t.Run(tc.name, func(t *testing.T) { - ctrl := gomock.NewController(t) - pieceStore := mock.NewMockPieceStore(ctrl) - sectorAccessor := mock.NewMockSectorAccessor(ctrl) - retrievalProv := mock.NewMockRetrievalProvider(ctrl) - ss, err := NewShardSelector(ctx, pieceStore, sectorAccessor, retrievalProv) - require.NoError(t, err) - - blockCid := testutil.GenerateCid() - require.NoError(t, err) - sk1cid := testutil.GenerateCid() - sk1 := shard.KeyFromCID(sk1cid) - shards := []shard.Key{sk1} - - pi := piecestore.PieceInfo{ - PieceCID: testutil.GenerateCid(), - Deals: tc.deals, - } - pieceStore.EXPECT().GetPieceInfo(sk1cid).AnyTimes().Return(pi, nil) - - for _, dl := range tc.deals { - isUnsealed := tc.isUnsealed[dl.SectorID] - sectorAccessor.EXPECT().IsUnsealed(gomock.Any(), dl.SectorID, gomock.Any(), gomock.Any()).AnyTimes().Return(isUnsealed, nil) - } - - ask := retrievalmarket.Ask{PricePerByte: abi.NewTokenAmount(tc.pricePerByte)} - retrievalProv.EXPECT().GetDynamicAsk(gomock.Any(), gomock.Any(), gomock.Any()).AnyTimes().Return(ask, nil) - - sk, err := ss.ShardSelectorF(blockCid, shards) - if tc.expectErr != nil { - require.ErrorIs(t, err, tc.expectErr) - } else { - require.NoError(t, err) - require.Equal(t, sk1, sk) - } - }) - } -} - -func TestShardSelectorCache(t *testing.T) { - ctx := context.Background() - - ctrl := gomock.NewController(t) - pieceStore := mock.NewMockPieceStore(ctrl) - sectorAccessor := mock.NewMockSectorAccessor(ctrl) - retrievalProv := mock.NewMockRetrievalProvider(ctrl) - ss, err := NewShardSelector(ctx, pieceStore, sectorAccessor, retrievalProv) - require.NoError(t, err) - - blockCid := testutil.GenerateCid() - require.NoError(t, err) - sk1cid := testutil.GenerateCid() - sk1 := shard.KeyFromCID(sk1cid) - shards := []shard.Key{sk1} - - pi := piecestore.PieceInfo{ - PieceCID: testutil.GenerateCid(), - Deals: []piecestore.DealInfo{{SectorID: 1}}, - } - pieceStore.EXPECT().GetPieceInfo(sk1cid). - // Expect there to be only one call to GetPieceInfo because after the first - // call the result should be cached - MinTimes(1).MaxTimes(1). - Return(pi, nil) - - sectorAccessor.EXPECT().IsUnsealed(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).AnyTimes().Return(true, nil) - - ask := retrievalmarket.Ask{PricePerByte: abi.NewTokenAmount(0)} - retrievalProv.EXPECT().GetDynamicAsk(gomock.Any(), gomock.Any(), gomock.Any()).AnyTimes().Return(ask, nil) - - var wg errgroup.Group - for i := 0; i < 100; i++ { - wg.Go(func() error { - _, err := ss.ShardSelectorF(blockCid, shards) - return err - }) - } - require.NoError(t, wg.Wait()) -} diff --git a/retrievalmarket/lp2pimpl/transports.go b/retrievalmarket/lp2pimpl/transports.go index 6fc0828b1..897ce2cab 100644 --- a/retrievalmarket/lp2pimpl/transports.go +++ b/retrievalmarket/lp2pimpl/transports.go @@ -5,7 +5,7 @@ import ( "fmt" "time" - "github.com/filecoin-project/boost-gfm/shared" + "github.com/filecoin-project/boost/markets/shared" "github.com/filecoin-project/boost/retrievalmarket/types" "github.com/filecoin-project/boost/safe" logging "github.com/ipfs/go-log/v2" diff --git a/retrievalmarket/mock/gen.go b/retrievalmarket/mock/gen.go deleted file mode 100644 index 60a1a8419..000000000 --- a/retrievalmarket/mock/gen.go +++ /dev/null @@ -1,4 +0,0 @@ -package mock - -//go:generate go run github.com/golang/mock/mockgen -destination=./piecestore.go -package=mock github.com/filecoin-project/go-fil-markets/piecestore PieceStore -//go:generate go run github.com/golang/mock/mockgen -destination=./retrievalmarket.go -package=mock github.com/filecoin-project/go-fil-markets/retrievalmarket RetrievalProvider,SectorAccessor diff --git a/retrievalmarket/mock/piecestore.go b/retrievalmarket/mock/piecestore.go deleted file mode 100644 index daadaf346..000000000 --- a/retrievalmarket/mock/piecestore.go +++ /dev/null @@ -1,152 +0,0 @@ -// Code generated by MockGen. DO NOT EDIT. -// Source: github.com/filecoin-project/go-fil-markets/piecestore (interfaces: PieceStore) - -// Package mock is a generated GoMock package. -package mock - -import ( - context "context" - reflect "reflect" - - piecestore "github.com/filecoin-project/boost-gfm/piecestore" - shared "github.com/filecoin-project/boost-gfm/shared" - gomock "github.com/golang/mock/gomock" - cid "github.com/ipfs/go-cid" -) - -// MockPieceStore is a mock of PieceStore interface. -type MockPieceStore struct { - ctrl *gomock.Controller - recorder *MockPieceStoreMockRecorder -} - -// MockPieceStoreMockRecorder is the mock recorder for MockPieceStore. -type MockPieceStoreMockRecorder struct { - mock *MockPieceStore -} - -// NewMockPieceStore creates a new mock instance. -func NewMockPieceStore(ctrl *gomock.Controller) *MockPieceStore { - mock := &MockPieceStore{ctrl: ctrl} - mock.recorder = &MockPieceStoreMockRecorder{mock} - return mock -} - -// EXPECT returns an object that allows the caller to indicate expected use. -func (m *MockPieceStore) EXPECT() *MockPieceStoreMockRecorder { - return m.recorder -} - -// AddDealForPiece mocks base method. -func (m *MockPieceStore) AddDealForPiece(arg0, arg1 cid.Cid, arg2 piecestore.DealInfo) error { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "AddDealForPiece", arg0, arg1, arg2) - ret0, _ := ret[0].(error) - return ret0 -} - -// AddDealForPiece indicates an expected call of AddDealForPiece. -func (mr *MockPieceStoreMockRecorder) AddDealForPiece(arg0, arg1, arg2 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "AddDealForPiece", reflect.TypeOf((*MockPieceStore)(nil).AddDealForPiece), arg0, arg1, arg2) -} - -// AddPieceBlockLocations mocks base method. -func (m *MockPieceStore) AddPieceBlockLocations(arg0 cid.Cid, arg1 map[cid.Cid]piecestore.BlockLocation) error { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "AddPieceBlockLocations", arg0, arg1) - ret0, _ := ret[0].(error) - return ret0 -} - -// AddPieceBlockLocations indicates an expected call of AddPieceBlockLocations. -func (mr *MockPieceStoreMockRecorder) AddPieceBlockLocations(arg0, arg1 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "AddPieceBlockLocations", reflect.TypeOf((*MockPieceStore)(nil).AddPieceBlockLocations), arg0, arg1) -} - -// GetCIDInfo mocks base method. -func (m *MockPieceStore) GetCIDInfo(arg0 cid.Cid) (piecestore.CIDInfo, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "GetCIDInfo", arg0) - ret0, _ := ret[0].(piecestore.CIDInfo) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// GetCIDInfo indicates an expected call of GetCIDInfo. -func (mr *MockPieceStoreMockRecorder) GetCIDInfo(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetCIDInfo", reflect.TypeOf((*MockPieceStore)(nil).GetCIDInfo), arg0) -} - -// GetPieceInfo mocks base method. -func (m *MockPieceStore) GetPieceInfo(arg0 cid.Cid) (piecestore.PieceInfo, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "GetPieceInfo", arg0) - ret0, _ := ret[0].(piecestore.PieceInfo) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// GetPieceInfo indicates an expected call of GetPieceInfo. -func (mr *MockPieceStoreMockRecorder) GetPieceInfo(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetPieceInfo", reflect.TypeOf((*MockPieceStore)(nil).GetPieceInfo), arg0) -} - -// ListCidInfoKeys mocks base method. -func (m *MockPieceStore) ListCidInfoKeys() ([]cid.Cid, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "ListCidInfoKeys") - ret0, _ := ret[0].([]cid.Cid) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// ListCidInfoKeys indicates an expected call of ListCidInfoKeys. -func (mr *MockPieceStoreMockRecorder) ListCidInfoKeys() *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "ListCidInfoKeys", reflect.TypeOf((*MockPieceStore)(nil).ListCidInfoKeys)) -} - -// ListPieceInfoKeys mocks base method. -func (m *MockPieceStore) ListPieceInfoKeys() ([]cid.Cid, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "ListPieceInfoKeys") - ret0, _ := ret[0].([]cid.Cid) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// ListPieceInfoKeys indicates an expected call of ListPieceInfoKeys. -func (mr *MockPieceStoreMockRecorder) ListPieceInfoKeys() *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "ListPieceInfoKeys", reflect.TypeOf((*MockPieceStore)(nil).ListPieceInfoKeys)) -} - -// OnReady mocks base method. -func (m *MockPieceStore) OnReady(arg0 shared.ReadyFunc) { - m.ctrl.T.Helper() - m.ctrl.Call(m, "OnReady", arg0) -} - -// OnReady indicates an expected call of OnReady. -func (mr *MockPieceStoreMockRecorder) OnReady(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "OnReady", reflect.TypeOf((*MockPieceStore)(nil).OnReady), arg0) -} - -// Start mocks base method. -func (m *MockPieceStore) Start(arg0 context.Context) error { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "Start", arg0) - ret0, _ := ret[0].(error) - return ret0 -} - -// Start indicates an expected call of Start. -func (mr *MockPieceStoreMockRecorder) Start(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Start", reflect.TypeOf((*MockPieceStore)(nil).Start), arg0) -} diff --git a/retrievalmarket/mock/retrievalmarket.go b/retrievalmarket/mock/retrievalmarket.go deleted file mode 100644 index fcc04cf05..000000000 --- a/retrievalmarket/mock/retrievalmarket.go +++ /dev/null @@ -1,229 +0,0 @@ -// Code generated by MockGen. DO NOT EDIT. -// Source: github.com/filecoin-project/go-fil-markets/retrievalmarket (interfaces: RetrievalProvider,SectorAccessor) - -// Package mock is a generated GoMock package. -package mock - -import ( - context "context" - io "io" - reflect "reflect" - - retrievalmarket "github.com/filecoin-project/boost-gfm/retrievalmarket" - shared "github.com/filecoin-project/boost-gfm/shared" - abi "github.com/filecoin-project/go-state-types/abi" - gomock "github.com/golang/mock/gomock" -) - -// MockRetrievalProvider is a mock of RetrievalProvider interface. -type MockRetrievalProvider struct { - ctrl *gomock.Controller - recorder *MockRetrievalProviderMockRecorder -} - -// MockRetrievalProviderMockRecorder is the mock recorder for MockRetrievalProvider. -type MockRetrievalProviderMockRecorder struct { - mock *MockRetrievalProvider -} - -// NewMockRetrievalProvider creates a new mock instance. -func NewMockRetrievalProvider(ctrl *gomock.Controller) *MockRetrievalProvider { - mock := &MockRetrievalProvider{ctrl: ctrl} - mock.recorder = &MockRetrievalProviderMockRecorder{mock} - return mock -} - -// EXPECT returns an object that allows the caller to indicate expected use. -func (m *MockRetrievalProvider) EXPECT() *MockRetrievalProviderMockRecorder { - return m.recorder -} - -// GetAsk mocks base method. -func (m *MockRetrievalProvider) GetAsk() *retrievalmarket.Ask { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "GetAsk") - ret0, _ := ret[0].(*retrievalmarket.Ask) - return ret0 -} - -// GetAsk indicates an expected call of GetAsk. -func (mr *MockRetrievalProviderMockRecorder) GetAsk() *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetAsk", reflect.TypeOf((*MockRetrievalProvider)(nil).GetAsk)) -} - -// GetDynamicAsk mocks base method. -func (m *MockRetrievalProvider) GetDynamicAsk(arg0 context.Context, arg1 retrievalmarket.PricingInput, arg2 []abi.DealID) (retrievalmarket.Ask, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "GetDynamicAsk", arg0, arg1, arg2) - ret0, _ := ret[0].(retrievalmarket.Ask) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// GetDynamicAsk indicates an expected call of GetDynamicAsk. -func (mr *MockRetrievalProviderMockRecorder) GetDynamicAsk(arg0, arg1, arg2 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetDynamicAsk", reflect.TypeOf((*MockRetrievalProvider)(nil).GetDynamicAsk), arg0, arg1, arg2) -} - -// ListDeals mocks base method. -func (m *MockRetrievalProvider) ListDeals() map[retrievalmarket.ProviderDealIdentifier]retrievalmarket.ProviderDealState { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "ListDeals") - ret0, _ := ret[0].(map[retrievalmarket.ProviderDealIdentifier]retrievalmarket.ProviderDealState) - return ret0 -} - -// ListDeals indicates an expected call of ListDeals. -func (mr *MockRetrievalProviderMockRecorder) ListDeals() *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "ListDeals", reflect.TypeOf((*MockRetrievalProvider)(nil).ListDeals)) -} - -// OnReady mocks base method. -func (m *MockRetrievalProvider) OnReady(arg0 shared.ReadyFunc) { - m.ctrl.T.Helper() - m.ctrl.Call(m, "OnReady", arg0) -} - -// OnReady indicates an expected call of OnReady. -func (mr *MockRetrievalProviderMockRecorder) OnReady(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "OnReady", reflect.TypeOf((*MockRetrievalProvider)(nil).OnReady), arg0) -} - -// SetAsk mocks base method. -func (m *MockRetrievalProvider) SetAsk(arg0 *retrievalmarket.Ask) { - m.ctrl.T.Helper() - m.ctrl.Call(m, "SetAsk", arg0) -} - -// SetAsk indicates an expected call of SetAsk. -func (mr *MockRetrievalProviderMockRecorder) SetAsk(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "SetAsk", reflect.TypeOf((*MockRetrievalProvider)(nil).SetAsk), arg0) -} - -// Start mocks base method. -func (m *MockRetrievalProvider) Start(arg0 context.Context) error { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "Start", arg0) - ret0, _ := ret[0].(error) - return ret0 -} - -// Start indicates an expected call of Start. -func (mr *MockRetrievalProviderMockRecorder) Start(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Start", reflect.TypeOf((*MockRetrievalProvider)(nil).Start), arg0) -} - -// Stop mocks base method. -func (m *MockRetrievalProvider) Stop() error { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "Stop") - ret0, _ := ret[0].(error) - return ret0 -} - -// Stop indicates an expected call of Stop. -func (mr *MockRetrievalProviderMockRecorder) Stop() *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Stop", reflect.TypeOf((*MockRetrievalProvider)(nil).Stop)) -} - -// SubscribeToEvents mocks base method. -func (m *MockRetrievalProvider) SubscribeToEvents(arg0 retrievalmarket.ProviderSubscriber) retrievalmarket.Unsubscribe { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "SubscribeToEvents", arg0) - ret0, _ := ret[0].(retrievalmarket.Unsubscribe) - return ret0 -} - -// SubscribeToEvents indicates an expected call of SubscribeToEvents. -func (mr *MockRetrievalProviderMockRecorder) SubscribeToEvents(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "SubscribeToEvents", reflect.TypeOf((*MockRetrievalProvider)(nil).SubscribeToEvents), arg0) -} - -// SubscribeToQueryEvents mocks base method. -func (m *MockRetrievalProvider) SubscribeToQueryEvents(arg0 retrievalmarket.ProviderQueryEventSubscriber) retrievalmarket.Unsubscribe { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "SubscribeToQueryEvents", arg0) - ret0, _ := ret[0].(retrievalmarket.Unsubscribe) - return ret0 -} - -// SubscribeToQueryEvents indicates an expected call of SubscribeToQueryEvents. -func (mr *MockRetrievalProviderMockRecorder) SubscribeToQueryEvents(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "SubscribeToQueryEvents", reflect.TypeOf((*MockRetrievalProvider)(nil).SubscribeToQueryEvents), arg0) -} - -// SubscribeToValidationEvents mocks base method. -func (m *MockRetrievalProvider) SubscribeToValidationEvents(arg0 retrievalmarket.ProviderValidationSubscriber) retrievalmarket.Unsubscribe { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "SubscribeToValidationEvents", arg0) - ret0, _ := ret[0].(retrievalmarket.Unsubscribe) - return ret0 -} - -// SubscribeToValidationEvents indicates an expected call of SubscribeToValidationEvents. -func (mr *MockRetrievalProviderMockRecorder) SubscribeToValidationEvents(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "SubscribeToValidationEvents", reflect.TypeOf((*MockRetrievalProvider)(nil).SubscribeToValidationEvents), arg0) -} - -// MockSectorAccessor is a mock of SectorAccessor interface. -type MockSectorAccessor struct { - ctrl *gomock.Controller - recorder *MockSectorAccessorMockRecorder -} - -// MockSectorAccessorMockRecorder is the mock recorder for MockSectorAccessor. -type MockSectorAccessorMockRecorder struct { - mock *MockSectorAccessor -} - -// NewMockSectorAccessor creates a new mock instance. -func NewMockSectorAccessor(ctrl *gomock.Controller) *MockSectorAccessor { - mock := &MockSectorAccessor{ctrl: ctrl} - mock.recorder = &MockSectorAccessorMockRecorder{mock} - return mock -} - -// EXPECT returns an object that allows the caller to indicate expected use. -func (m *MockSectorAccessor) EXPECT() *MockSectorAccessorMockRecorder { - return m.recorder -} - -// IsUnsealed mocks base method. -func (m *MockSectorAccessor) IsUnsealed(arg0 context.Context, arg1 abi.SectorNumber, arg2, arg3 abi.UnpaddedPieceSize) (bool, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "IsUnsealed", arg0, arg1, arg2, arg3) - ret0, _ := ret[0].(bool) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// IsUnsealed indicates an expected call of IsUnsealed. -func (mr *MockSectorAccessorMockRecorder) IsUnsealed(arg0, arg1, arg2, arg3 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "IsUnsealed", reflect.TypeOf((*MockSectorAccessor)(nil).IsUnsealed), arg0, arg1, arg2, arg3) -} - -// UnsealSector mocks base method. -func (m *MockSectorAccessor) UnsealSector(arg0 context.Context, arg1 abi.SectorNumber, arg2, arg3 abi.UnpaddedPieceSize) (io.ReadCloser, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "UnsealSector", arg0, arg1, arg2, arg3) - ret0, _ := ret[0].(io.ReadCloser) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// UnsealSector indicates an expected call of UnsealSector. -func (mr *MockSectorAccessorMockRecorder) UnsealSector(arg0, arg1, arg2, arg3 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "UnsealSector", reflect.TypeOf((*MockSectorAccessor)(nil).UnsealSector), arg0, arg1, arg2, arg3) -} diff --git a/retrievalmarket/rtvllog/db.go b/retrievalmarket/rtvllog/db.go index 5a89b79d8..5ee646a63 100644 --- a/retrievalmarket/rtvllog/db.go +++ b/retrievalmarket/rtvllog/db.go @@ -8,8 +8,8 @@ import ( "strings" "time" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - datatransfer "github.com/filecoin-project/go-data-transfer" + datatransfer2 "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/go-state-types/big" "github.com/ipfs/go-cid" @@ -32,8 +32,8 @@ type RetrievalDealState struct { UpdatedAt time.Time LocalPeerID peer.ID PeerID peer.ID - DealID retrievalmarket.DealID - TransferID datatransfer.TransferID + DealID legacyretrievaltypes.DealID + TransferID datatransfer2.TransferID PayloadCID cid.Cid PieceCID *cid.Cid PaymentInterval uint64 @@ -278,7 +278,7 @@ func (d *RetrievalLogDB) Count(ctx context.Context, isIndexer *bool) (int, error return count, err } -func (d *RetrievalLogDB) Update(ctx context.Context, state retrievalmarket.ProviderDealState) error { +func (d *RetrievalLogDB) Update(ctx context.Context, state legacyretrievaltypes.ProviderDealState) error { fields := map[string]interface{}{ "Status": state.Status.String(), "TotalSent": state.TotalSent, @@ -297,7 +297,7 @@ func (d *RetrievalLogDB) Update(ctx context.Context, state retrievalmarket.Provi return d.update(ctx, fields, where, args...) } -func (d *RetrievalLogDB) UpdateDataTransferState(ctx context.Context, event datatransfer.Event, state datatransfer.ChannelState) error { +func (d *RetrievalLogDB) UpdateDataTransferState(ctx context.Context, event datatransfer2.Event, state datatransfer2.ChannelState) error { peerID := state.OtherPeer().String() transferID := state.TransferID() if err := d.insertDTEvent(ctx, peerID, transferID, event); err != nil { @@ -305,7 +305,7 @@ func (d *RetrievalLogDB) UpdateDataTransferState(ctx context.Context, event data } fields := map[string]interface{}{ - "DTStatus": datatransfer.Statuses[state.Status()], + "DTStatus": datatransfer2.Statuses[state.Status()], "DTMessage": state.Message(), "UpdatedAt": time.Now(), } @@ -326,10 +326,10 @@ func (d *RetrievalLogDB) update(ctx context.Context, fields map[string]interface return err } -func (d *RetrievalLogDB) insertDTEvent(ctx context.Context, peerID string, transferID datatransfer.TransferID, event datatransfer.Event) error { +func (d *RetrievalLogDB) insertDTEvent(ctx context.Context, peerID string, transferID datatransfer2.TransferID, event datatransfer2.Event) error { qry := "INSERT INTO RetrievalDataTransferEvents (PeerID, TransferID, CreatedAt, Name, Message) " + "VALUES (?, ?, ?, ?, ?)" - _, err := d.db.ExecContext(ctx, qry, peerID, transferID, event.Timestamp, datatransfer.Events[event.Code], event.Message) + _, err := d.db.ExecContext(ctx, qry, peerID, transferID, event.Timestamp, datatransfer2.Events[event.Code], event.Message) return err } @@ -339,7 +339,7 @@ type DTEvent struct { Message string } -func (d *RetrievalLogDB) ListDTEvents(ctx context.Context, peerID string, transferID datatransfer.TransferID) ([]DTEvent, error) { +func (d *RetrievalLogDB) ListDTEvents(ctx context.Context, peerID string, transferID datatransfer2.TransferID) ([]DTEvent, error) { qry := "SELECT CreatedAt, Name, Message " + "FROM RetrievalDataTransferEvents " + "WHERE PeerID = ? AND TransferID = ? " + @@ -372,10 +372,10 @@ func (d *RetrievalLogDB) ListDTEvents(ctx context.Context, peerID string, transf return dtEvents, nil } -func (d *RetrievalLogDB) InsertMarketsEvent(ctx context.Context, event retrievalmarket.ProviderEvent, state retrievalmarket.ProviderDealState) error { +func (d *RetrievalLogDB) InsertMarketsEvent(ctx context.Context, event legacyretrievaltypes.ProviderEvent, state legacyretrievaltypes.ProviderDealState) error { // Ignore block sent events as we are recording the equivalent event for // data-transfer, and it's a high-frequency event - if event == retrievalmarket.ProviderEventBlockSent { + if event == legacyretrievaltypes.ProviderEventBlockSent { return nil } @@ -385,7 +385,7 @@ func (d *RetrievalLogDB) InsertMarketsEvent(ctx context.Context, event retrieval state.Receiver.String(), state.ID, time.Now(), - retrievalmarket.ProviderEvents[event], + legacyretrievaltypes.ProviderEvents[event], state.Status.String(), state.Message) return err @@ -398,7 +398,7 @@ type MarketEvent struct { Message string } -func (d *RetrievalLogDB) ListMarketEvents(ctx context.Context, peerID string, dealID retrievalmarket.DealID) ([]MarketEvent, error) { +func (d *RetrievalLogDB) ListMarketEvents(ctx context.Context, peerID string, dealID legacyretrievaltypes.DealID) ([]MarketEvent, error) { qry := "SELECT CreatedAt, Name, Status, Message " + "FROM RetrievalMarketEvents " + "WHERE PeerID = ? AND DealID = ? " + diff --git a/retrievalmarket/rtvllog/retrieval_log.go b/retrievalmarket/rtvllog/retrieval_log.go index 3711bf577..c1c0941be 100644 --- a/retrievalmarket/rtvllog/retrieval_log.go +++ b/retrievalmarket/rtvllog/retrieval_log.go @@ -2,14 +2,12 @@ package rtvllog import ( "context" - "errors" "sync" "time" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/boost/node/modules/dtypes" + datatransfer2 "github.com/filecoin-project/boost/datatransfer" "github.com/filecoin-project/boost/retrievalmarket/server" - datatransfer "github.com/filecoin-project/go-data-transfer" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" logging "github.com/ipfs/go-log/v2" ) @@ -18,7 +16,6 @@ var log = logging.Logger("rtrvlog") type RetrievalLog struct { db *RetrievalLogDB duration time.Duration - dataTransfer dtypes.ProviderDataTransfer gsur *server.GraphsyncUnpaidRetrieval stalledTimeout time.Duration ctx context.Context @@ -29,7 +26,7 @@ type RetrievalLog struct { lastUpdate map[string]time.Time } -func NewRetrievalLog(db *RetrievalLogDB, duration time.Duration, dt dtypes.ProviderDataTransfer, stalledTimeout time.Duration, gsur *server.GraphsyncUnpaidRetrieval) *RetrievalLog { +func NewRetrievalLog(db *RetrievalLogDB, duration time.Duration, stalledTimeout time.Duration, gsur *server.GraphsyncUnpaidRetrieval) *RetrievalLog { if duration < stalledTimeout { log.Warnf("the RetrievalLogDuration (%s) should exceed the StalledRetrievalTimeout (%s)", duration.String(), stalledTimeout.String()) } @@ -37,7 +34,6 @@ func NewRetrievalLog(db *RetrievalLogDB, duration time.Duration, dt dtypes.Provi return &RetrievalLog{ db: db, duration: duration, - dataTransfer: dt, gsur: gsur, stalledTimeout: stalledTimeout, dbUpdates: make(chan func(), 256), @@ -49,12 +45,11 @@ func (r *RetrievalLog) Start(ctx context.Context) { r.ctx = ctx go r.gcUpdateMap(ctx) go r.gcDatabase(ctx) - go r.gcRetrievals(ctx) go r.processDBUpdates(ctx) } // Called when there is a retrieval ask query -func (r *RetrievalLog) OnQueryEvent(evt retrievalmarket.ProviderQueryEvent) { +func (r *RetrievalLog) OnQueryEvent(evt legacyretrievaltypes.ProviderQueryEvent) { log.Debugw("query-event", "status", evt.Response.Status, "msg", evt.Response.Message, @@ -71,10 +66,10 @@ func (r *RetrievalLog) OnQueryEvent(evt retrievalmarket.ProviderQueryEvent) { st.Message = evt.Error.Error() } } else { - if evt.Response.Status == retrievalmarket.QueryResponseUnavailable { + if evt.Response.Status == legacyretrievaltypes.QueryResponseUnavailable { st.Status = "unavailable" } - if evt.Response.Status == retrievalmarket.QueryResponseError { + if evt.Response.Status == legacyretrievaltypes.QueryResponseError { st.Status = "errored" } } @@ -92,10 +87,10 @@ func (r *RetrievalLog) OnQueryEvent(evt retrievalmarket.ProviderQueryEvent) { // This occurs when the client makes a graphsync retrieval request, and the // Storage Provider validates the request (eg checking its parameters for // validity, checking for acceptance against the retrieval filter, etc) -func (r *RetrievalLog) OnValidationEvent(evt retrievalmarket.ProviderValidationEvent) { +func (r *RetrievalLog) OnValidationEvent(evt legacyretrievaltypes.ProviderValidationEvent) { // Ignore ErrPause and ErrResume because they are signalling errors, not // actual errors because of incorrect behaviour. - if evt.Error == nil || evt.Error == datatransfer.ErrPause || evt.Error == datatransfer.ErrResume { + if evt.Error == nil || evt.Error == datatransfer2.ErrPause || evt.Error == datatransfer2.ErrResume { return } @@ -103,7 +98,7 @@ func (r *RetrievalLog) OnValidationEvent(evt retrievalmarket.ProviderValidationE st := &RetrievalDealState{ PeerID: evt.Receiver, PayloadCID: evt.BaseCid, - Status: retrievalmarket.DealStatusErrored.String(), + Status: legacyretrievaltypes.DealStatusErrored.String(), Message: evt.Error.Error(), } if evt.Response != nil { @@ -132,10 +127,10 @@ func (r *RetrievalLog) OnValidationEvent(evt retrievalmarket.ProviderValidationE } // Called when there is an event from the data-transfer subsystem -func (r *RetrievalLog) OnDataTransferEvent(event datatransfer.Event, state datatransfer.ChannelState) { +func (r *RetrievalLog) OnDataTransferEvent(event datatransfer2.Event, state datatransfer2.ChannelState) { log.Debugw("dt-event", - "evt", datatransfer.Events[event.Code], - "status", datatransfer.Statuses[state.Status()], + "evt", datatransfer2.Events[event.Code], + "status", datatransfer2.Statuses[state.Status()], "message", state.Message(), "is-pull", state.IsPull()) @@ -145,10 +140,10 @@ func (r *RetrievalLog) OnDataTransferEvent(event datatransfer.Event, state datat } switch event.Code { - case datatransfer.DataQueued, datatransfer.DataQueuedProgress, datatransfer.DataSentProgress, - datatransfer.DataReceived, datatransfer.DataReceivedProgress: + case datatransfer2.DataQueued, datatransfer2.DataQueuedProgress, datatransfer2.DataSentProgress, + datatransfer2.DataReceived, datatransfer2.DataReceivedProgress: return - case datatransfer.DataSent: + case datatransfer2.DataSent: // To prevent too frequent updates, only allow data sent updates if it's // been more than half a second since the last one if !r.allowUpdate(state.ChannelID().String()) { @@ -165,29 +160,29 @@ func (r *RetrievalLog) OnDataTransferEvent(event datatransfer.Event, state datat } // Called when there is a markets event -func (r *RetrievalLog) OnRetrievalEvent(event retrievalmarket.ProviderEvent, state retrievalmarket.ProviderDealState) { +func (r *RetrievalLog) OnRetrievalEvent(event legacyretrievaltypes.ProviderEvent, state legacyretrievaltypes.ProviderDealState) { // To prevent too frequent updates, only allow block sent updates if it's // been more than half a second since the last one - if event == retrievalmarket.ProviderEventBlockSent && !r.allowUpdate(state.ChannelID.String()) { + if event == legacyretrievaltypes.ProviderEventBlockSent && !r.allowUpdate(state.ChannelID.String()) { return } - var transferID datatransfer.TransferID + var transferID datatransfer2.TransferID if state.ChannelID != nil { log.Debugw("event", - "evt", retrievalmarket.ProviderEvents[event], + "evt", legacyretrievaltypes.ProviderEvents[event], "status", state.Status, "initiator", state.ChannelID.Initiator, "responder", state.ChannelID.Responder, "transfer id", state.ChannelID.ID) transferID = state.ChannelID.ID } else { - log.Debugw("event", "evt", retrievalmarket.ProviderEvents[event], "status", state.Status) + log.Debugw("event", "evt", legacyretrievaltypes.ProviderEvents[event], "status", state.Status) } r.dbUpdate(func() { var err error - if event == retrievalmarket.ProviderEventOpen { + if event == legacyretrievaltypes.ProviderEventOpen { err = r.db.Insert(r.ctx, &RetrievalDealState{ PeerID: state.Receiver, DealID: state.ID, @@ -278,58 +273,6 @@ func (r *RetrievalLog) gcDatabase(ctx context.Context) { } } -// Periodically cancels stalled retrievals older than 30mins -func (r *RetrievalLog) gcRetrievals(ctx context.Context) { - ticker := time.NewTicker(5 * time.Minute) - defer ticker.Stop() - - for { - select { - case <-ctx.Done(): - return - case now := <-ticker.C: - // Get retrievals last updated - rows, err := r.db.ListLastUpdatedAndOpen(ctx, now.Add(-r.stalledTimeout)) - - if err != nil { - log.Errorw("error fetching open, stalled retrievals", "err", err) - continue - } - - var wg sync.WaitGroup - for _, row := range rows { - if row.TransferID <= 0 { - continue - } - wg.Add(1) - go func(s RetrievalDealState) { - // Don't wait for more than 5 seconds for the cancel - // message to be sent when cancelling an unpaid retrieval - unpaidRtrvCtx, cancel := context.WithTimeout(ctx, time.Second*5) - defer cancel() - defer wg.Done() - - // Try to cancel an unpaid retrieval with the given transfer id first - err := r.gsur.CancelTransfer(unpaidRtrvCtx, s.TransferID, &s.PeerID) - if err != nil && errors.Is(err, server.ErrRetrievalNotFound) { - // Couldn't find an unpaid retrieval with that id, try - // to cancel a legacy, paid retrieval - chid := datatransfer.ChannelID{Initiator: s.PeerID, Responder: s.LocalPeerID, ID: s.TransferID} - err = r.dataTransfer.CloseDataTransferChannel(ctx, chid) - } - - if err != nil { - log.Debugw("error canceling retrieval", "dealID", s.DealID, "err", err) - } else { - log.Infof("Canceled retrieval %s, older than %s", s.DealID, r.stalledTimeout) - } - }(row) - } - wg.Wait() - } - } -} - // Perform database updates in a separate thread so that they don't block the // event publisher loop func (r *RetrievalLog) dbUpdate(update func()) { diff --git a/retrievalmarket/server/channelstate.go b/retrievalmarket/server/channelstate.go index 1f1b48ed8..159fe1ba9 100644 --- a/retrievalmarket/server/channelstate.go +++ b/retrievalmarket/server/channelstate.go @@ -3,9 +3,9 @@ package server import ( "bytes" - "github.com/filecoin-project/boost-gfm/retrievalmarket" graphsync "github.com/filecoin-project/boost-graphsync" - datatransfer "github.com/filecoin-project/go-data-transfer" + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" "github.com/ipfs/go-cid" "github.com/ipld/go-ipld-prime" "github.com/ipld/go-ipld-prime/codec/dagcbor" @@ -22,12 +22,12 @@ const RetrievalTypeLegs RetrievalType = "Legs" type retrievalState struct { retType RetrievalType cs *channelState - mkts *retrievalmarket.ProviderDealState + mkts *legacyretrievaltypes.ProviderDealState gsReq graphsync.RequestID } -func (r retrievalState) ChannelState() channelState { return *r.cs } -func (r retrievalState) ProviderDealState() retrievalmarket.ProviderDealState { return *r.mkts } +func (r retrievalState) ChannelState() channelState { return *r.cs } +func (r retrievalState) ProviderDealState() legacyretrievaltypes.ProviderDealState { return *r.mkts } // channelState is immutable channel data plus mutable state type channelState struct { @@ -68,6 +68,26 @@ type channelState struct { message string } +func (c channelState) Vouchers() []datatransfer.Voucher { + //TODO implement me + panic("implement me") +} + +func (c channelState) VoucherResults() []datatransfer.VoucherResult { + //TODO implement me + panic("implement me") +} + +func (c channelState) LastVoucher() datatransfer.Voucher { + //TODO implement me + panic("implement me") +} + +func (c channelState) LastVoucherResult() datatransfer.VoucherResult { + //TODO implement me + panic("implement me") +} + // EmptyChannelState is the zero value for channel state, meaning not present var EmptyChannelState = channelState{} @@ -149,22 +169,6 @@ func (c channelState) Message() string { return c.message } -func (c channelState) Vouchers() []datatransfer.Voucher { - return nil -} - -func (c channelState) LastVoucher() datatransfer.Voucher { - return nil -} - -func (c channelState) LastVoucherResult() datatransfer.VoucherResult { - return nil -} - -func (c channelState) VoucherResults() []datatransfer.VoucherResult { - return nil -} - func (c channelState) SelfPeer() peer.ID { return c.selfPeer } diff --git a/node/modules/provider_data_transfer.go b/retrievalmarket/server/datatransfer.go similarity index 81% rename from node/modules/provider_data_transfer.go rename to retrievalmarket/server/datatransfer.go index a27b3f657..dc841edd5 100644 --- a/node/modules/provider_data_transfer.go +++ b/retrievalmarket/server/datatransfer.go @@ -1,13 +1,14 @@ -package modules +package server import ( "context" "errors" "time" + "github.com/filecoin-project/boost/datatransfer" + dtimpl "github.com/filecoin-project/boost/datatransfer/impl" marketevents "github.com/filecoin-project/boost/markets/loggers" "github.com/filecoin-project/boost/node/modules/dtypes" - dtimpl "github.com/filecoin-project/go-data-transfer/impl" lotus_dtypes "github.com/filecoin-project/lotus/node/modules/dtypes" "github.com/filecoin-project/lotus/node/repo" "github.com/ipfs/go-datastore" @@ -15,8 +16,10 @@ import ( "go.uber.org/fx" ) +type ProviderDataTransfer datatransfer.Manager + // NewProviderDataTransfer returns a data transfer manager -func NewProviderDataTransfer(lc fx.Lifecycle, net dtypes.ProviderTransferNetwork, transport dtypes.ProviderTransport, ds lotus_dtypes.MetadataDS, r repo.LockedRepo) (dtypes.ProviderDataTransfer, error) { +func NewProviderDataTransfer(lc fx.Lifecycle, net dtypes.ProviderTransferNetwork, transport dtypes.ProviderTransport, ds lotus_dtypes.MetadataDS, r repo.LockedRepo) (ProviderDataTransfer, error) { dtDs := namespace.Wrap(ds, datastore.NewKey("/datatransfer/provider/transfers")) dt, err := dtimpl.NewDataTransfer(dtDs, net, transport) @@ -28,6 +31,7 @@ func NewProviderDataTransfer(lc fx.Lifecycle, net dtypes.ProviderTransferNetwork lc.Append(fx.Hook{ OnStart: func(ctx context.Context) error { dt.SubscribeToEvents(marketevents.DataTransferLogger) + log.Infof("started provider data transfer") return dt.Start(ctx) }, OnStop: func(ctx context.Context) error { diff --git a/retrievalmarket/server/events.go b/retrievalmarket/server/events.go index a8d1efb23..fea7db765 100644 --- a/retrievalmarket/server/events.go +++ b/retrievalmarket/server/events.go @@ -2,23 +2,24 @@ package server import ( "errors" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - datatransfer "github.com/filecoin-project/go-data-transfer" - "github.com/hannahhoward/go-pubsub" "time" + + datatransfer2 "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" + "github.com/hannahhoward/go-pubsub" ) -func (g *GraphsyncUnpaidRetrieval) SubscribeToDataTransferEvents(subscriber datatransfer.Subscriber) datatransfer.Unsubscribe { - return datatransfer.Unsubscribe(g.pubSubDT.Subscribe(subscriber)) +func (g *GraphsyncUnpaidRetrieval) SubscribeToDataTransferEvents(subscriber datatransfer2.Subscriber) datatransfer2.Unsubscribe { + return datatransfer2.Unsubscribe(g.pubSubDT.Subscribe(subscriber)) } type dtEvent struct { - evt datatransfer.Event - state datatransfer.ChannelState + evt datatransfer2.Event + state datatransfer2.ChannelState } -func (g *GraphsyncUnpaidRetrieval) publishDTEvent(evtCode datatransfer.EventCode, msg string, chst datatransfer.ChannelState) { - evt := datatransfer.Event{ +func (g *GraphsyncUnpaidRetrieval) publishDTEvent(evtCode datatransfer2.EventCode, msg string, chst datatransfer2.ChannelState) { + evt := datatransfer2.Event{ Code: evtCode, Message: msg, Timestamp: time.Now(), @@ -34,7 +35,7 @@ func eventDispatcherDT(evt pubsub.Event, subscriberFn pubsub.SubscriberFn) error if !ok { return errors.New("wrong type of event") } - cb, ok := subscriberFn.(datatransfer.Subscriber) + cb, ok := subscriberFn.(datatransfer2.Subscriber) if !ok { return errors.New("wrong type of subscriber function") } @@ -42,16 +43,16 @@ func eventDispatcherDT(evt pubsub.Event, subscriberFn pubsub.SubscriberFn) error return nil } -func (g *GraphsyncUnpaidRetrieval) SubscribeToMarketsEvents(subscriber retrievalmarket.ProviderSubscriber) retrievalmarket.Unsubscribe { - return retrievalmarket.Unsubscribe(g.pubSubMkts.Subscribe(subscriber)) +func (g *GraphsyncUnpaidRetrieval) SubscribeToMarketsEvents(subscriber ProviderSubscriber) legacyretrievaltypes.Unsubscribe { + return legacyretrievaltypes.Unsubscribe(g.pubSubMkts.Subscribe(subscriber)) } type mktsEvent struct { - evt retrievalmarket.ProviderEvent - state retrievalmarket.ProviderDealState + evt legacyretrievaltypes.ProviderEvent + state legacyretrievaltypes.ProviderDealState } -func (g *GraphsyncUnpaidRetrieval) publishMktsEvent(evt retrievalmarket.ProviderEvent, state retrievalmarket.ProviderDealState) { +func (g *GraphsyncUnpaidRetrieval) publishMktsEvent(evt legacyretrievaltypes.ProviderEvent, state legacyretrievaltypes.ProviderDealState) { err := g.pubSubMkts.Publish(mktsEvent{evt: evt, state: state}) if err != nil { log.Warnf("err publishing markets event: %s", err.Error()) @@ -63,10 +64,19 @@ func eventDispatcherMkts(evt pubsub.Event, subscriberFn pubsub.SubscriberFn) err if !ok { return errors.New("wrong type of event") } - cb, ok := subscriberFn.(retrievalmarket.ProviderSubscriber) + cb, ok := subscriberFn.(ProviderSubscriber) if !ok { return errors.New("wrong type of event") } cb(ie.evt, ie.state) return nil } + +// ProviderSubscriber is a callback that is registered to listen for retrieval events on a provider +type ProviderSubscriber func(event legacyretrievaltypes.ProviderEvent, state legacyretrievaltypes.ProviderDealState) + +// ProviderQueryEventSubscriber is a callback that is registered to listen for query message events +type ProviderQueryEventSubscriber func(evt legacyretrievaltypes.ProviderQueryEvent) + +// ProviderValidationSubscriber is a callback that is registered to listen for validation events +type ProviderValidationSubscriber func(evt legacyretrievaltypes.ProviderValidationEvent) diff --git a/retrievalmarket/server/gsunpaidretrieval.go b/retrievalmarket/server/gsunpaidretrieval.go index e9b76b388..5ebea3236 100644 --- a/retrievalmarket/server/gsunpaidretrieval.go +++ b/retrievalmarket/server/gsunpaidretrieval.go @@ -6,30 +6,29 @@ import ( "fmt" "sync" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - retrievalimpl "github.com/filecoin-project/boost-gfm/retrievalmarket/impl" - "github.com/filecoin-project/boost-gfm/retrievalmarket/migrations" graphsync "github.com/filecoin-project/boost-graphsync" + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/datatransfer/encoding" + "github.com/filecoin-project/boost/datatransfer/message" + "github.com/filecoin-project/boost/datatransfer/network" + "github.com/filecoin-project/boost/datatransfer/registry" + "github.com/filecoin-project/boost/datatransfer/transport/graphsync/extension" "github.com/filecoin-project/boost/metrics" "github.com/filecoin-project/boost/piecedirectory" - "github.com/filecoin-project/boost/retrievalmarket/types" - datatransfer "github.com/filecoin-project/go-data-transfer" - "github.com/filecoin-project/go-data-transfer/encoding" - "github.com/filecoin-project/go-data-transfer/message" - "github.com/filecoin-project/go-data-transfer/network" - "github.com/filecoin-project/go-data-transfer/registry" - "github.com/filecoin-project/go-data-transfer/transport/graphsync/extension" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes/migrations" "github.com/filecoin-project/go-state-types/abi" "github.com/hannahhoward/go-pubsub" logging "github.com/ipfs/go-log/v2" - "github.com/ipld/go-ipld-prime" "github.com/libp2p/go-libp2p/core/peer" cbg "github.com/whyrusleeping/cbor-gen" "go.opencensus.io/stats" ) -var log = logging.Logger("boostgs") -var ErrRetrievalNotFound = fmt.Errorf("no transfer found") +var ( + log = logging.Logger("boostgs") + ErrRetrievalNotFound = fmt.Errorf("no transfer found") +) var incomingReqExtensions = []graphsync.ExtensionName{ extension.ExtensionIncomingRequest1_1, @@ -42,10 +41,6 @@ type reqId struct { id datatransfer.TransferID } -type LinkSystemProvider interface { - LinkSys() *ipld.LinkSystem -} - // GraphsyncUnpaidRetrieval intercepts incoming requests to Graphsync. // If the request is for a paid retrieval, it is forwarded to the existing // Graphsync implementation. @@ -58,7 +53,6 @@ type GraphsyncUnpaidRetrieval struct { validator *requestValidator pubSubDT *pubsub.PubSub pubSubMkts *pubsub.PubSub - linkSystem LinkSystemProvider activeRetrievalsLk sync.RWMutex activeRetrievals map[reqId]*retrievalState @@ -75,16 +69,37 @@ var defaultExtensions = []graphsync.ExtensionName{ extension.ExtensionDataTransfer1_1, } +type RetrievalAskGetter interface { + GetAsk() *legacyretrievaltypes.Ask +} + +type retrievalAskGetter struct { + ask legacyretrievaltypes.Ask +} + +func (rag *retrievalAskGetter) GetAsk() *legacyretrievaltypes.Ask { + return &rag.ask +} + +func NewRetrievalAskGetter() RetrievalAskGetter { + return &retrievalAskGetter{ + ask: legacyretrievaltypes.Ask{ + PricePerByte: abi.NewTokenAmount(0), + UnsealPrice: abi.NewTokenAmount(0), + }, + } +} + type ValidationDeps struct { - DealDecider retrievalimpl.DealDecider + DealDecider DealDecider PieceDirectory *piecedirectory.PieceDirectory SectorAccessor SectorAccessor - AskStore AskGetter + AskStore RetrievalAskGetter } -func NewGraphsyncUnpaidRetrieval(peerID peer.ID, gs graphsync.GraphExchange, dtnet network.DataTransferNetwork, vdeps ValidationDeps, ls LinkSystemProvider) (*GraphsyncUnpaidRetrieval, error) { +func NewGraphsyncUnpaidRetrieval(peerID peer.ID, gs graphsync.GraphExchange, dtnet network.DataTransferNetwork, vdeps ValidationDeps) (*GraphsyncUnpaidRetrieval, error) { typeRegistry := registry.NewRegistry() - err := typeRegistry.Register(&retrievalmarket.DealProposal{}, nil) + err := typeRegistry.Register(&legacyretrievaltypes.DealProposal{}, nil) if err != nil { return nil, err } @@ -92,11 +107,6 @@ func NewGraphsyncUnpaidRetrieval(peerID peer.ID, gs graphsync.GraphExchange, dtn if err != nil { return nil, err } - err = typeRegistry.Register(&types.LegsVoucherDTv1{}, nil) - if err != nil { - return nil, err - } - return &GraphsyncUnpaidRetrieval{ GraphExchange: gs, peerID: peerID, @@ -106,7 +116,6 @@ func NewGraphsyncUnpaidRetrieval(peerID peer.ID, gs graphsync.GraphExchange, dtn pubSubMkts: pubsub.New(eventDispatcherMkts), validator: newRequestValidator(vdeps), activeRetrievals: make(map[reqId]*retrievalState), - linkSystem: ls, }, nil } @@ -114,16 +123,6 @@ func (g *GraphsyncUnpaidRetrieval) Start(ctx context.Context) error { g.ctx = ctx g.validator.ctx = ctx - if g.linkSystem != nil && g.linkSystem.LinkSys() != nil { - // The index provider uses graphsync to fetch advertisements. - // We need to tell graphsync to use a different IPLD Link System to provide - // the advertisements (instead of using the blockstore). - err := g.RegisterPersistenceOption("indexstore", *g.linkSystem.LinkSys()) - if err != nil { - return fmt.Errorf("setting persistence option for index advertisement retrieval: %w", err) - } - } - return nil } @@ -214,6 +213,7 @@ func (g *GraphsyncUnpaidRetrieval) List() []retrievalState { // Called when a transfer is received by graphsync and queued for processing func (g *GraphsyncUnpaidRetrieval) RegisterIncomingRequestQueuedHook(hook graphsync.OnIncomingRequestQueuedHook) graphsync.UnregisterHookFunc { return g.GraphExchange.RegisterIncomingRequestQueuedHook(func(p peer.ID, request graphsync.RequestData, hookActions graphsync.RequestQueuedHookActions) { + log.Debugw("incoming request queued", "request", request) stats.Record(g.ctx, metrics.GraphsyncRequestQueuedCount.M(1)) interceptRtvl, err := g.interceptRetrieval(p, request) @@ -240,12 +240,14 @@ func (g *GraphsyncUnpaidRetrieval) interceptRetrieval(p peer.ID, request graphsy } // Extension not found, ignore if msg == nil { + log.Debugw("no extension found", "request", request) return false, nil } // When a data transfer request comes in on graphsync, the remote peer // initiated a pull request for data. If it's not a request, ignore it. if !msg.IsRequest() { + log.Debugw("ignoring non-request message", "request", request) return false, nil } @@ -256,49 +258,40 @@ func (g *GraphsyncUnpaidRetrieval) interceptRetrieval(p peer.ID, request graphsy // be in our map (because we must have already processed the new // retrieval request) _, ok := g.isActiveUnpaidRetrieval(reqId{p: p, id: msg.TransferID()}) + log.Debugw("ignoring non-new request", "request", request, "isActiveUnpaidRetrieval", ok) return ok, nil } // The request is for a new transfer / restart transfer, so check if it's - // for an unpaid retrieval + // for an unpaid retrieval. We are explicitly checking for voucher type to be + // legacyretrievaltypes.DealProposal{}. Rest are all rejected at this stage. voucher, decodeErr := g.decodeVoucher(dtRequest, g.decoder) if decodeErr != nil { // If we don't recognize the voucher, don't intercept the retrieval. // Instead it will be passed through to the legacy code for processing. + log.Debugw("decoding new request voucher", "request", request, "err", decodeErr) if !errors.Is(decodeErr, unknownVoucherErr) { return false, fmt.Errorf("decoding new request voucher: %w", decodeErr) } } - switch v := voucher.(type) { - case *types.LegsVoucherDTv1: - // This is a go-legs voucher (used by the network indexer to retrieve - // deal announcements) - - // Treat it the same way as a retrieval deal proposal with no payment - params, err := retrievalmarket.NewParamsV1(abi.NewTokenAmount(0), 0, 0, request.Selector(), nil, abi.NewTokenAmount(0)) - if err != nil { - return false, err - } - proposal := retrievalmarket.DealProposal{ - PayloadCID: request.Root(), - Params: params, - } - return g.handleRetrievalDeal(p, msg, proposal, request, RetrievalTypeLegs) - case *retrievalmarket.DealProposal: + case *legacyretrievaltypes.DealProposal: // This is a retrieval deal proposal := *v + log.Debugw("intercepting retrieval deal", "proposal", proposal) return g.handleRetrievalDeal(p, msg, proposal, request, RetrievalTypeDeal) case *migrations.DealProposal0: // This is a retrieval deal with an older format proposal := migrations.MigrateDealProposal0To1(*v) + log.Debugw("intercepting retrieval deal v1", "proposal", proposal) return g.handleRetrievalDeal(p, msg, proposal, request, RetrievalTypeDeal) } + log.Debugw("ignoring request", "request", request) return false, nil } -func (g *GraphsyncUnpaidRetrieval) handleRetrievalDeal(peerID peer.ID, msg datatransfer.Message, proposal retrievalmarket.DealProposal, request graphsync.RequestData, retType RetrievalType) (bool, error) { +func (g *GraphsyncUnpaidRetrieval) handleRetrievalDeal(peerID peer.ID, msg datatransfer.Message, proposal legacyretrievaltypes.DealProposal, request graphsync.RequestData, retType RetrievalType) (bool, error) { // If it's a paid retrieval, do not intercept it if !proposal.UnsealPrice.IsZero() || !proposal.PricePerByte.IsZero() { return false, nil @@ -320,10 +313,10 @@ func (g *GraphsyncUnpaidRetrieval) handleRetrievalDeal(peerID peer.ID, msg datat isPull: true, } - mktsState := &retrievalmarket.ProviderDealState{ + mktsState := &legacyretrievaltypes.ProviderDealState{ DealProposal: proposal, ChannelID: &datatransfer.ChannelID{ID: msg.TransferID(), Initiator: peerID, Responder: g.peerID}, - Status: retrievalmarket.DealStatusNew, + Status: legacyretrievaltypes.DealStatusNew, Receiver: peerID, FundsReceived: abi.NewTokenAmount(0), } @@ -349,6 +342,7 @@ func (g *GraphsyncUnpaidRetrieval) handleRetrievalDeal(peerID peer.ID, msg datat // Called by graphsync when an incoming request is processed func (g *GraphsyncUnpaidRetrieval) RegisterIncomingRequestHook(hook graphsync.OnIncomingRequestHook) graphsync.UnregisterHookFunc { return g.GraphExchange.RegisterIncomingRequestHook(func(p peer.ID, request graphsync.RequestData, hookActions graphsync.IncomingRequestHookActions) { + log.Debugw("incoming request", "request", request) stats.Record(g.ctx, metrics.GraphsyncRequestStartedCount.M(1)) // Check if this is a request for a retrieval that we should handle @@ -357,6 +351,7 @@ func (g *GraphsyncUnpaidRetrieval) RegisterIncomingRequestHook(hook graphsync.On // Otherwise pass it through to the legacy code hook(p, request, hookActions) stats.Record(g.ctx, metrics.GraphsyncRequestStartedPaidCount.M(1)) + log.Debugw("passing paid request through to legacy code", "request", request) return } @@ -371,29 +366,21 @@ func (g *GraphsyncUnpaidRetrieval) RegisterIncomingRequestHook(hook graphsync.On if msg.IsRestart() { dtOpenMsg += " (restart)" } + log.Debugw("handling unpaid request", "request", request, "msg", msg, "state", state) g.publishDTEvent(datatransfer.Open, dtOpenMsg, state.cs) - g.publishMktsEvent(retrievalmarket.ProviderEventOpen, *state.mkts) + g.publishMktsEvent(legacyretrievaltypes.ProviderEventOpen, *state.mkts) err := func() error { voucher, decodeErr := g.decodeVoucher(msg, g.decoder) if decodeErr != nil { + log.Debugw("decoding new request voucher", "request", request, "err", decodeErr) return fmt.Errorf("decoding new request voucher: %w", decodeErr) } // Validate the request - var res datatransfer.VoucherResult - var validateErr error - - if _, ok := voucher.(*types.LegsVoucherDTv1); ok { - // It's a go-legs voucher, so we need to tell Graphsync to - // use a different IPLD Link System to serve the data (instead - // of using the regular blockstore) - res = &types.LegsVoucherResultDtv1{} - validateErr = nil - hookActions.UsePersistenceOption("indexstore") - } else { - res, validateErr = g.validator.validatePullRequest(msg.IsRestart(), p, voucher, request.Root(), request.Selector()) - } + res, validateErr := g.validator.validatePullRequest(msg.IsRestart(), p, voucher, request.Root(), request.Selector()) + log.Debugw("validating request", "request", request, "result", res, "err", validateErr) + isAccepted := validateErr == nil const isPaused = false // There are no payments required, so never pause resultType := datatransfer.EmptyTypeIdentifier @@ -425,8 +412,8 @@ func (g *GraphsyncUnpaidRetrieval) RegisterIncomingRequestHook(hook graphsync.On return validateErr }() - if err != nil { + log.Debugw("validation failed", "request", request, "err", err) hookActions.TerminateWithError(err) g.failTransfer(state, err) stats.Record(g.ctx, metrics.GraphsyncRequestStartedUnpaidFailCount.M(1)) @@ -439,12 +426,13 @@ func (g *GraphsyncUnpaidRetrieval) RegisterIncomingRequestHook(hook graphsync.On // Fire events state.cs.status = datatransfer.Ongoing g.publishDTEvent(datatransfer.Accept, "", state.cs) - state.mkts.Status = retrievalmarket.DealStatusUnsealing - g.publishMktsEvent(retrievalmarket.ProviderEventDealAccepted, *state.mkts) - state.mkts.Status = retrievalmarket.DealStatusUnsealed - g.publishMktsEvent(retrievalmarket.ProviderEventUnsealComplete, *state.mkts) + state.mkts.Status = legacyretrievaltypes.DealStatusUnsealing + g.publishMktsEvent(legacyretrievaltypes.ProviderEventDealAccepted, *state.mkts) + state.mkts.Status = legacyretrievaltypes.DealStatusUnsealed + g.publishMktsEvent(legacyretrievaltypes.ProviderEventUnsealComplete, *state.mkts) stats.Record(g.ctx, metrics.GraphsyncRequestStartedUnpaidSuccessCount.M(1)) + log.Debugw("successfully validated request", "request", request) }) } @@ -498,28 +486,18 @@ func (g *GraphsyncUnpaidRetrieval) RegisterCompletedResponseListener(listener gr } // Fire markets blocks completed event - state.mkts.Status = retrievalmarket.DealStatusBlocksComplete - g.publishMktsEvent(retrievalmarket.ProviderEventBlocksCompleted, *state.mkts) + state.mkts.Status = legacyretrievaltypes.DealStatusBlocksComplete + g.publishMktsEvent(legacyretrievaltypes.ProviderEventBlocksCompleted, *state.mkts) // Include a markets protocol Completed message in the response - var voucherResult encoding.Encodable - var voucherType datatransfer.TypeIdentifier - if state.retType == RetrievalTypeDeal { - dealResponse := &retrievalmarket.DealResponse{ - ID: state.mkts.DealProposal.ID, - Status: retrievalmarket.DealStatusCompleted, - } - voucherResult = dealResponse - voucherType = dealResponse.Type() - } else { - legsResponse := &types.LegsVoucherResultDtv1{} - voucherResult = legsResponse - voucherType = legsResponse.Type() + dealResponse := &legacyretrievaltypes.DealResponse{ + ID: state.mkts.DealProposal.ID, + Status: legacyretrievaltypes.DealStatusCompleted, } const isAccepted = true const isPaused = false - respMsg, err := message.CompleteResponse(msg.TransferID(), isAccepted, isPaused, voucherType, voucherResult) + respMsg, err := message.CompleteResponse(msg.TransferID(), isAccepted, isPaused, dealResponse.Type(), dealResponse) if err != nil { g.failTransfer(state, fmt.Errorf("getting complete response: %w", err)) return @@ -536,8 +514,8 @@ func (g *GraphsyncUnpaidRetrieval) RegisterCompletedResponseListener(listener gr state.cs.status = datatransfer.Completed g.publishDTEvent(datatransfer.Complete, "", state.cs) // Fire markets blocks completed event - state.mkts.Status = retrievalmarket.DealStatusCompleted - g.publishMktsEvent(retrievalmarket.ProviderEventComplete, *state.mkts) + state.mkts.Status = legacyretrievaltypes.DealStatusCompleted + g.publishMktsEvent(legacyretrievaltypes.ProviderEventComplete, *state.mkts) stats.Record(g.ctx, metrics.GraphsyncRequestCompletedUnpaidSuccessCount.M(1)) log.Infow("successfully sent completion message to requestor", "peer", p) @@ -546,7 +524,6 @@ func (g *GraphsyncUnpaidRetrieval) RegisterCompletedResponseListener(listener gr func (g *GraphsyncUnpaidRetrieval) RegisterRequestorCancelledListener(listener graphsync.OnRequestorCancelledListener) graphsync.UnregisterHookFunc { return g.GraphExchange.RegisterRequestorCancelledListener(func(p peer.ID, request graphsync.RequestData) { - stats.Record(g.ctx, metrics.GraphsyncRequestClientCancelledCount.M(1)) _, state, intercept := g.isRequestForActiveUnpaidRetrieval(p, request) @@ -558,8 +535,8 @@ func (g *GraphsyncUnpaidRetrieval) RegisterRequestorCancelledListener(listener g state.cs.status = datatransfer.Cancelled g.publishDTEvent(datatransfer.Cancel, "client cancelled", state.cs) - state.mkts.Status = retrievalmarket.DealStatusCancelled - g.publishMktsEvent(retrievalmarket.ProviderEventCancelComplete, *state.mkts) + state.mkts.Status = legacyretrievaltypes.DealStatusCancelled + g.publishMktsEvent(legacyretrievaltypes.ProviderEventCancelComplete, *state.mkts) g.untrackTransfer(p, state.cs.transferID) @@ -630,8 +607,8 @@ func (g *GraphsyncUnpaidRetrieval) failTransfer(state *retrievalState, err error state.cs.status = datatransfer.Failed state.cs.message = err.Error() g.publishDTEvent(datatransfer.Error, err.Error(), state.cs) - state.mkts.Status = retrievalmarket.DealStatusErrored - g.publishMktsEvent(retrievalmarket.ProviderEventDataTransferError, *state.mkts) + state.mkts.Status = legacyretrievaltypes.DealStatusErrored + g.publishMktsEvent(legacyretrievaltypes.ProviderEventDataTransferError, *state.mkts) g.untrackTransfer(state.cs.recipient, state.cs.transferID) log.Infow("transfer failed", "transfer id", state.cs.transferID, "peer", state.cs.recipient, "err", err) @@ -682,6 +659,6 @@ func (g *GraphsyncUnpaidRetrieval) isActiveUnpaidRetrieval(id reqId) (*retrieval return state, ok } -func (g *GraphsyncUnpaidRetrieval) SubscribeToValidationEvents(sub retrievalmarket.ProviderValidationSubscriber) retrievalmarket.Unsubscribe { +func (g *GraphsyncUnpaidRetrieval) SubscribeToValidationEvents(sub ProviderValidationSubscriber) legacyretrievaltypes.Unsubscribe { return g.validator.Subscribe(sub) } diff --git a/retrievalmarket/server/gsunpaidretrieval_test.go b/retrievalmarket/server/gsunpaidretrieval_test.go deleted file mode 100644 index cb8d5b6bb..000000000 --- a/retrievalmarket/server/gsunpaidretrieval_test.go +++ /dev/null @@ -1,384 +0,0 @@ -package server - -import ( - "context" - "errors" - "fmt" - "io" - "os" - "testing" - "time" - - "github.com/filecoin-project/boost-gfm/retrievalmarket" - retrievalimpl "github.com/filecoin-project/boost-gfm/retrievalmarket/impl" - "github.com/filecoin-project/boost-gfm/retrievalmarket/impl/askstore" - "github.com/filecoin-project/boost-gfm/retrievalmarket/impl/testnodes" - rmnet "github.com/filecoin-project/boost-gfm/retrievalmarket/network" - tut "github.com/filecoin-project/boost-gfm/shared_testutil" - graphsyncimpl "github.com/filecoin-project/boost-graphsync/impl" - "github.com/filecoin-project/boost-graphsync/network" - "github.com/filecoin-project/boost-graphsync/storeutil" - bdclientutil "github.com/filecoin-project/boost/extern/boostd-data/clientutil" - "github.com/filecoin-project/boost/extern/boostd-data/model" - "github.com/filecoin-project/boost/piecedirectory" - "github.com/filecoin-project/go-address" - datatransfer "github.com/filecoin-project/go-data-transfer" - dtimpl "github.com/filecoin-project/go-data-transfer/impl" - "github.com/filecoin-project/go-data-transfer/testutil" - dtgstransport "github.com/filecoin-project/go-data-transfer/transport/graphsync" - "github.com/filecoin-project/go-state-types/abi" - "github.com/google/uuid" - "github.com/ipfs/go-cid" - "github.com/ipfs/go-datastore" - "github.com/ipfs/go-datastore/namespace" - logging "github.com/ipfs/go-log/v2" - "github.com/ipld/go-car/v2" - basicnode "github.com/ipld/go-ipld-prime/node/basic" - "github.com/ipld/go-ipld-prime/traversal/selector" - "github.com/ipld/go-ipld-prime/traversal/selector/builder" - "github.com/stretchr/testify/require" -) - -var tlog = logging.Logger("testgs") - -type testCase struct { - name string - reqPayloadCid cid.Cid - watch func(client retrievalmarket.RetrievalClient, gsupr *GraphsyncUnpaidRetrieval) - ask *retrievalmarket.Ask - noUnsealedCopy bool - useCarV2 bool - expectErr bool - expectClientCancelEvent bool - expectProviderCancelEvent bool - expectRejection string -} - -var providerCancelled = errors.New("provider cancelled") -var clientCancelled = errors.New("client cancelled") -var clientRejected = errors.New("client received reject response") - -func TestGS(t *testing.T) { - //_ = logging.SetLogLevel("testgs", "debug") - _ = logging.SetLogLevel("testgs", "info") - //_ = logging.SetLogLevel("dt-impl", "debug") - - missingCid := cid.MustParse("baguqeeraaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa") - - testCases := []testCase{{ - name: "happy path", - }, { - name: "happy path w/ carv2", - useCarV2: true, - }, { - name: "request missing payload cid", - reqPayloadCid: missingCid, - expectErr: true, - }, { - name: "request for piece with no unsealed sectors", - noUnsealedCopy: true, - expectErr: true, - expectRejection: "no unsealed piece", - }, { - name: "request for non-zero price per byte", - ask: &retrievalmarket.Ask{ - UnsealPrice: abi.NewTokenAmount(0), - PricePerByte: abi.NewTokenAmount(1), - }, - expectErr: true, - expectRejection: "ask price is non-zero", - }, { - // Note: we disregard the unseal price because we only serve deals - // with an unsealed piece, so the unseal price is irrelevant. - // Therefore the retrieval should succeed for non-zero unseal price. - name: "request for non-zero unseal price", - ask: &retrievalmarket.Ask{ - UnsealPrice: abi.NewTokenAmount(1), - PricePerByte: abi.NewTokenAmount(0), - }, - }, { - name: "cancel request after sending 2 blocks", - watch: func(client retrievalmarket.RetrievalClient, gsupr *GraphsyncUnpaidRetrieval) { - count := 0 - gsupr.outgoingBlockHook = func(state *retrievalState) { - count++ - if count == 2 { - tlog.Debug("cancelling client deal") - err := client.CancelDeal(state.mkts.ID) - require.NoError(t, err) - } - if count == 10 { - tlog.Warn("sending last block but client cancel hasn't arrived yet") - } - } - }, - expectClientCancelEvent: true, - expectProviderCancelEvent: true, - }, { - name: "provider cancel request after sending 2 blocks", - watch: func(client retrievalmarket.RetrievalClient, gsupr *GraphsyncUnpaidRetrieval) { - count := 0 - gsupr.outgoingBlockHook = func(state *retrievalState) { - count++ - if count == 2 { - tlog.Debug("provider cancelling client deal") - err := gsupr.CancelTransfer(context.TODO(), state.cs.transferID, &state.cs.recipient) - require.NoError(t, err) - } - if count == 10 { - tlog.Warn("sending last block but client cancel hasn't arrived yet") - } - } - }, - expectErr: true, - expectClientCancelEvent: true, - }, { - name: "provider cancel request after sending 2 blocks without peer id", - watch: func(client retrievalmarket.RetrievalClient, gsupr *GraphsyncUnpaidRetrieval) { - count := 0 - gsupr.outgoingBlockHook = func(state *retrievalState) { - count++ - if count == 2 { - tlog.Debug("provider cancelling client deal") - err := gsupr.CancelTransfer(context.TODO(), state.cs.transferID, nil) - require.NoError(t, err) - } - if count == 10 { - tlog.Warn("sending last block but client cancel hasn't arrived yet") - } - } - }, - expectErr: true, - expectClientCancelEvent: true, - }} - - for _, tc := range testCases { - t.Run(tc.name, func(t *testing.T) { - runRequestTest(t, tc) - }) - } -} - -func runRequestTest(t *testing.T, tc testCase) { - ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) - defer cancel() - - // Create a CAR file and set up mocks - testData := tut.NewLibp2pTestData(ctx, t) - - // Create a random CAR file - carRootCid, carFilePath := piecedirectory.CreateCarFile(t) - - var sectionReader car.SectionReader - - if tc.useCarV2 { - var err error - sectionReader, err = os.Open(carFilePath) - require.NoError(t, err) - defer sectionReader.(*os.File).Close() - } else { - carReader, err := car.OpenReader(carFilePath) - require.NoError(t, err) - defer carReader.Close() - sectionReader, err = carReader.DataReader() - require.NoError(t, err) - - } - - // Any calls to get a reader over data should return a reader over the random CAR file - pr := piecedirectory.CreateMockPieceReader(t, sectionReader) - - carv1Bytes, err := io.ReadAll(sectionReader) - require.NoError(t, err) - carSize := len(carv1Bytes) - - maddr := address.TestAddress - pieceCid := tut.GenerateCids(1)[0] - sectorID := abi.SectorNumber(1) - offset := abi.PaddedPieceSize(0) - dealInfo := model.DealInfo{ - DealUuid: uuid.New().String(), - ChainDealID: abi.DealID(1), - MinerAddr: maddr, - SectorID: sectorID, - PieceOffset: offset, - PieceLength: abi.UnpaddedPieceSize(carSize).Padded(), - } - - askStore, err := askstore.NewAskStore(namespace.Wrap(testData.Ds1, datastore.NewKey("retrieval-ask")), datastore.NewKey("latest")) - require.NoError(t, err) - ask := &retrievalmarket.Ask{UnsealPrice: abi.NewTokenAmount(0), PricePerByte: abi.NewTokenAmount(0)} - if tc.ask != nil { - ask = tc.ask - } - err = askStore.SetAsk(ask) - require.NoError(t, err) - - cl := bdclientutil.NewTestStore(ctx) - defer cl.Close(ctx) - - pd := piecedirectory.NewPieceDirectory(cl, pr, 1) - pd.Start(ctx) - err = pd.AddDealForPiece(ctx, pieceCid, dealInfo) - require.NoError(t, err) - - vdeps := ValidationDeps{ - PieceDirectory: pd, - SectorAccessor: &mockSectorAccessor{ - unsealed: !tc.noUnsealedCopy, - }, - AskStore: askStore, - } - - // Create a blockstore over the CAR file blocks - carDataBs, err := pd.GetBlockstore(ctx, pieceCid) - require.NoError(t, err) - - // Wrap graphsync with the graphsync unpaid retrieval interceptor - linkSystem2 := storeutil.LinkSystemForBlockstore(carDataBs) - gs2 := graphsyncimpl.New(ctx, network.NewFromLibp2pHost(testData.Host2), linkSystem2) - gsupr, err := NewGraphsyncUnpaidRetrieval(testData.Host2.ID(), gs2, testData.DTNet2, vdeps, nil) - require.NoError(t, err) - - // Create a Graphsync transport and call SetEventHandler, which registers - // listeners for all the Graphsync hooks. - gsTransport := dtgstransport.NewTransport(testData.Host2.ID(), gsupr) - err = gsTransport.SetEventHandler(nil) - require.NoError(t, err) - - // Create the retrieval provider with the graphsync unpaid retrieval interceptor - paymentAddress := address.TestAddress2 - - gsupr.SubscribeToDataTransferEvents(func(event datatransfer.Event, channelState datatransfer.ChannelState) { - tlog.Debugf("prov dt: %s %s / %s", datatransfer.Events[event.Code], event.Message, datatransfer.Statuses[channelState.Status()]) - }) - err = gsupr.Start(ctx) - require.NoError(t, err) - - // Create a retrieval client - retrievalPeer := retrievalmarket.RetrievalPeer{ - Address: paymentAddress, - ID: testData.Host2.ID(), - } - retrievalClientNode := testnodes.NewTestRetrievalClientNode(testnodes.TestRetrievalClientNodeParams{}) - retrievalClientNode.ExpectKnownAddresses(retrievalPeer, nil) - client := createRetrievalClient(ctx, t, testData, retrievalClientNode) - tut.StartAndWaitForReady(ctx, t, client) - - if tc.watch != nil { - tc.watch(client, gsupr) - } - - // Watch for provider completion - providerResChan := make(chan error) - gsupr.SubscribeToMarketsEvents(func(event retrievalmarket.ProviderEvent, state retrievalmarket.ProviderDealState) { - tlog.Debugf("prov mkt: %s %s %s", retrievalmarket.ProviderEvents[event], state.Status.String(), state.Message) - switch event { - case retrievalmarket.ProviderEventComplete: - providerResChan <- nil - case retrievalmarket.ProviderEventCancelComplete: - providerResChan <- providerCancelled - case retrievalmarket.ProviderEventDataTransferError: - providerResChan <- errors.New(state.Message) - } - }) - - // Watch for client completion - clientResChan := make(chan error) - client.SubscribeToEvents(func(event retrievalmarket.ClientEvent, state retrievalmarket.ClientDealState) { - tlog.Debugf("clnt mkt: %s %s %s", event.String(), state.Status.String(), state.Message) - switch event { - case retrievalmarket.ClientEventComplete: - clientResChan <- nil - case retrievalmarket.ClientEventCancelComplete: - clientResChan <- clientCancelled - case retrievalmarket.ClientEventDealRejected: - clientResChan <- fmt.Errorf("%s :%w", state.Message, clientRejected) - case retrievalmarket.ClientEventDataTransferError: - clientResChan <- errors.New(state.Message) - } - }) - - // Retrieve the data - tlog.Infof("Retrieve cid %s from peer %s", carRootCid, retrievalPeer.ID) - // Use an explore-all but add unixfs-preload to make sure we have UnixFS - // ADL support wired up. - ssb := builder.NewSelectorSpecBuilder(basicnode.Prototype.Any) - sel := ssb.ExploreInterpretAs("unixfs-preload", ssb.ExploreRecursive( - selector.RecursionLimitNone(), - ssb.ExploreAll(ssb.ExploreRecursiveEdge()), - )).Node() - params, err := retrievalmarket.NewParamsV1(abi.NewTokenAmount(0), 0, 0, sel, nil, abi.NewTokenAmount(0)) - require.NoError(t, err) - if tc.reqPayloadCid != cid.Undef { - carRootCid = tc.reqPayloadCid - } - _, err = client.Retrieve(ctx, 1, carRootCid, params, abi.NewTokenAmount(0), retrievalPeer, address.TestAddress, address.TestAddress2) - require.NoError(t, err) - - // Wait for provider completion - err = waitFor(ctx, t, providerResChan) - if tc.expectErr || tc.expectProviderCancelEvent { - require.Error(t, err) - if tc.expectProviderCancelEvent { - require.EqualError(t, err, providerCancelled.Error()) - } - } else { - require.NoError(t, err) - } - - // Wait for client completion - err = waitFor(ctx, t, clientResChan) - if tc.expectErr || tc.expectClientCancelEvent { - require.Error(t, err) - if tc.expectClientCancelEvent { - require.EqualError(t, err, clientCancelled.Error()) - } else if tc.expectRejection != "" { - require.ErrorContains(t, err, tc.expectRejection) - } - } else { - require.NoError(t, err) - } - - // final verification -- the server has no active graphsync requests - stats := gsupr.GraphExchange.Stats() - require.Equal(t, stats.IncomingRequests.Active, uint64(0)) -} - -func createRetrievalClient(ctx context.Context, t *testing.T, testData *tut.Libp2pTestData, retrievalClientNode *testnodes.TestRetrievalClientNode) retrievalmarket.RetrievalClient { - nw1 := rmnet.NewFromLibp2pHost(testData.Host1, rmnet.RetryParameters(0, 0, 0, 0)) - gs1 := graphsyncimpl.New(ctx, network.NewFromLibp2pHost(testData.Host1), testData.LinkSystem1) - dtTransport1 := dtgstransport.NewTransport(testData.Host1.ID(), gs1) - dt1, err := dtimpl.NewDataTransfer(testData.DTStore1, testData.DTNet1, dtTransport1) - require.NoError(t, err) - testutil.StartAndWaitForReady(ctx, t, dt1) - require.NoError(t, err) - clientDs := namespace.Wrap(testData.Ds1, datastore.NewKey("/retrievals/client")) - ba := tut.NewTestRetrievalBlockstoreAccessor() - client, err := retrievalimpl.NewClient(nw1, dt1, retrievalClientNode, &tut.TestPeerResolver{}, clientDs, ba) - require.NoError(t, err) - - dt1.SubscribeToEvents(func(event datatransfer.Event, channelState datatransfer.ChannelState) { - tlog.Debugf("client dt: %s %s / %s", datatransfer.Events[event.Code], event.Message, datatransfer.Statuses[channelState.Status()]) - }) - - return client -} - -func waitFor(ctx context.Context, t *testing.T, resChan chan error) error { - var err error - select { - case <-ctx.Done(): - require.Fail(t, "test timed out") - case err = <-resChan: - } - return err -} - -type mockSectorAccessor struct { - unsealed bool -} - -func (m *mockSectorAccessor) IsUnsealed(ctx context.Context, minerAddr address.Address, sectorID abi.SectorNumber, offset abi.UnpaddedPieceSize, length abi.UnpaddedPieceSize) (bool, error) { - return m.unsealed, nil -} diff --git a/retrievalmarket/server/provider_pieces.go b/retrievalmarket/server/provider_pieces.go index a145f6ad4..084d95c0c 100644 --- a/retrievalmarket/server/provider_pieces.go +++ b/retrievalmarket/server/provider_pieces.go @@ -3,6 +3,7 @@ package server import ( "context" "fmt" + "github.com/filecoin-project/boost/extern/boostd-data/model" "github.com/filecoin-project/boost/piecedirectory" "github.com/filecoin-project/go-state-types/abi" @@ -13,11 +14,6 @@ import ( "github.com/multiformats/go-multihash" ) -// This code is copied directly from -// https://github.com/filecoin-project/go-fil-markets/blob/955fd43fad7da2e68539c257f0c8199a6b0c2a4d/retrievalmarket/impl/provider_pieces.go#L1 -// TODO: Create a PR against go-fil-markets to make these methods public, -// so that we can import them from go-fil-markets instead of copying the code here. - // MaxIdentityCIDBytes is the largest identity CID as a PayloadCID that we are // willing to decode const MaxIdentityCIDBytes = 2 << 10 diff --git a/retrievalmarket/server/queryask.go b/retrievalmarket/server/queryask.go index 8fa79c94a..ecb890371 100644 --- a/retrievalmarket/server/queryask.go +++ b/retrievalmarket/server/queryask.go @@ -6,8 +6,8 @@ import ( "fmt" "time" - "github.com/filecoin-project/boost-gfm/retrievalmarket" "github.com/filecoin-project/boost/piecedirectory" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" "github.com/filecoin-project/boost/safe" "github.com/filecoin-project/go-address" cborutil "github.com/filecoin-project/go-cbor-util" @@ -31,12 +31,12 @@ type QueryAskHandler struct { minerAddress address.Address pd *piecedirectory.PieceDirectory sa SectorAccessor - askStore AskGetter + askStore RetrievalAskGetter full v1api.FullNode host host.Host } -func NewQueryAskHandler(host host.Host, maddr address.Address, pd *piecedirectory.PieceDirectory, sa SectorAccessor, askStore AskGetter, full v1api.FullNode) *QueryAskHandler { +func NewQueryAskHandler(host host.Host, maddr address.Address, pd *piecedirectory.PieceDirectory, sa SectorAccessor, askStore RetrievalAskGetter, full v1api.FullNode) *QueryAskHandler { return &QueryAskHandler{ host: host, minerAddress: maddr, @@ -48,11 +48,11 @@ func NewQueryAskHandler(host host.Host, maddr address.Address, pd *piecedirector } func (qa *QueryAskHandler) Start() { - qa.host.SetStreamHandler(retrievalmarket.QueryProtocolID, safe.Handle(qa.HandleQueryStream)) + qa.host.SetStreamHandler(legacyretrievaltypes.QueryProtocolID, safe.Handle(qa.HandleQueryStream)) } func (qa *QueryAskHandler) Stop() { - qa.host.RemoveStreamHandler(retrievalmarket.QueryProtocolID) + qa.host.RemoveStreamHandler(legacyretrievaltypes.QueryProtocolID) } func (qa *QueryAskHandler) HandleQueryStream(stream network.Stream) { @@ -61,7 +61,7 @@ func (qa *QueryAskHandler) HandleQueryStream(stream network.Stream) { // Set a deadline on reading from the stream so it doesn't hang _ = stream.SetReadDeadline(time.Now().Add(providerReadDeadline)) - var query retrievalmarket.Query + var query legacyretrievaltypes.Query err := query.UnmarshalCBOR(stream) _ = stream.SetReadDeadline(time.Time{}) // Clear read deadline so conn doesn't get closed if err != nil { @@ -73,13 +73,13 @@ func (qa *QueryAskHandler) HandleQueryStream(stream network.Stream) { defer cancel() answer, err := qa.getQueryResponse(ctx, query) if err != nil { - status := retrievalmarket.QueryResponseError - if errors.Is(err, retrievalmarket.ErrNotFound) { - status = retrievalmarket.QueryResponseUnavailable + status := legacyretrievaltypes.QueryResponseError + if errors.Is(err, legacyretrievaltypes.ErrNotFound) { + status = legacyretrievaltypes.QueryResponseUnavailable } - answer = &retrievalmarket.QueryResponse{ + answer = &legacyretrievaltypes.QueryResponse{ Status: status, - PieceCIDFound: retrievalmarket.QueryItemUnavailable, + PieceCIDFound: legacyretrievaltypes.QueryItemUnavailable, PaymentAddress: qa.minerAddress, MinPricePerByte: big.Zero(), UnsealPrice: big.Zero(), @@ -96,7 +96,7 @@ func (qa *QueryAskHandler) HandleQueryStream(stream network.Stream) { } } -func (qa *QueryAskHandler) getQueryResponse(ctx context.Context, query retrievalmarket.Query) (*retrievalmarket.QueryResponse, error) { +func (qa *QueryAskHandler) getQueryResponse(ctx context.Context, query legacyretrievaltypes.Query) (*legacyretrievaltypes.QueryResponse, error) { // Fetch the payment address the client should send the payment to head, err := qa.full.ChainHead(ctx) if err != nil { @@ -122,10 +122,10 @@ func (qa *QueryAskHandler) getQueryResponse(ctx context.Context, query retrieval pieceInfo, _ := GetBestPieceInfoMatch(ctx, qa.sa, pieces, pieceCID) if !pieceInfo.PieceCID.Defined() { - if piecesErr != nil && !errors.Is(piecesErr, retrievalmarket.ErrNotFound) { + if piecesErr != nil && !errors.Is(piecesErr, legacyretrievaltypes.ErrNotFound) { return nil, fmt.Errorf("fetching piece to retrieve from: %w", piecesErr) } - return nil, fmt.Errorf("getting pieces for payload cid %s: %w", query.PayloadCID, retrievalmarket.ErrNotFound) + return nil, fmt.Errorf("getting pieces for payload cid %s: %w", query.PayloadCID, legacyretrievaltypes.ErrNotFound) } if len(pieceInfo.Deals) == 0 { @@ -145,11 +145,11 @@ func (qa *QueryAskHandler) getQueryResponse(ctx context.Context, query retrieval return nil, errors.New("no ask configured in ask-store") } - return &retrievalmarket.QueryResponse{ + return &legacyretrievaltypes.QueryResponse{ PaymentAddress: minerInfo.Worker, - Status: retrievalmarket.QueryResponseAvailable, + Status: legacyretrievaltypes.QueryResponseAvailable, Size: uint64(pieceInfo.Deals[0].PieceLength.Unpadded()), - PieceCIDFound: retrievalmarket.QueryItemAvailable, + PieceCIDFound: legacyretrievaltypes.QueryItemAvailable, MinPricePerByte: currAsk.PricePerByte, MaxPaymentInterval: currAsk.PaymentInterval, MaxPaymentIntervalIncrease: currAsk.PaymentIntervalIncrease, diff --git a/retrievalmarket/server/types.go b/retrievalmarket/server/types.go index 786064ac3..8e1791db3 100644 --- a/retrievalmarket/server/types.go +++ b/retrievalmarket/server/types.go @@ -2,15 +2,15 @@ package server import ( "context" - "github.com/filecoin-project/boost-gfm/retrievalmarket" + + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" "github.com/filecoin-project/go-address" "github.com/filecoin-project/go-state-types/abi" ) -type AskGetter interface { - GetAsk() *retrievalmarket.Ask -} - type SectorAccessor interface { IsUnsealed(ctx context.Context, minerAddr address.Address, sectorID abi.SectorNumber, offset abi.UnpaddedPieceSize, length abi.UnpaddedPieceSize) (bool, error) } + +// DealDecider is a function that makes a decision about whether to accept a deal +type DealDecider func(ctx context.Context, state legacyretrievaltypes.ProviderDealState) (bool, string, error) diff --git a/retrievalmarket/server/validation.go b/retrievalmarket/server/validation.go index d0116075a..27727aefe 100644 --- a/retrievalmarket/server/validation.go +++ b/retrievalmarket/server/validation.go @@ -6,9 +6,10 @@ import ( "errors" "fmt" - "github.com/filecoin-project/boost-gfm/retrievalmarket" - "github.com/filecoin-project/boost-gfm/retrievalmarket/migrations" - datatransfer "github.com/filecoin-project/go-data-transfer" + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes/migrations" + + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" "github.com/hannahhoward/go-pubsub" "github.com/ipfs/go-cid" "github.com/ipld/go-ipld-prime" @@ -39,19 +40,20 @@ func newRequestValidator(vdeps ValidationDeps) *requestValidator { // request to pull data or a new request created when the data transfer is // restarted (eg after a connection failure). func (rv *requestValidator) validatePullRequest(isRestart bool, receiver peer.ID, voucher datatransfer.Voucher, baseCid cid.Cid, selector ipld.Node) (datatransfer.VoucherResult, error) { - proposal, ok := voucher.(*retrievalmarket.DealProposal) + proposal, ok := voucher.(*legacyretrievaltypes.DealProposal) var legacyProtocol bool if !ok { - legacyProposal, ok := voucher.(*migrations.DealProposal0) - if !ok { - return nil, errors.New("wrong voucher type") - } - newProposal := migrations.MigrateDealProposal0To1(*legacyProposal) - proposal = &newProposal - legacyProtocol = true + return nil, errors.New("wrong voucher type") + //legacyProposal, ok := voucher.(*migrations.DealProposal0) + //if !ok { + // return nil, errors.New("wrong voucher type") + //} + //newProposal := migrations.MigrateDealProposal0To1(*legacyProposal) + //proposal = &newProposal + //legacyProtocol = true } response, err := rv.validatePull(receiver, proposal, legacyProtocol, baseCid, selector) - _ = rv.psub.Publish(retrievalmarket.ProviderValidationEvent{ + _ = rv.psub.Publish(legacyretrievaltypes.ProviderValidationEvent{ IsRestart: isRestart, Receiver: receiver, Proposal: proposal, @@ -72,16 +74,16 @@ func (rv *requestValidator) validatePullRequest(isRestart bool, receiver peer.ID return &response, err } -func (rv *requestValidator) validatePull(receiver peer.ID, proposal *retrievalmarket.DealProposal, legacyProtocol bool, baseCid cid.Cid, selector ipld.Node) (retrievalmarket.DealResponse, error) { - response := retrievalmarket.DealResponse{ +func (rv *requestValidator) validatePull(receiver peer.ID, proposal *legacyretrievaltypes.DealProposal, legacyProtocol bool, baseCid cid.Cid, selector ipld.Node) (legacyretrievaltypes.DealResponse, error) { + response := legacyretrievaltypes.DealResponse{ ID: proposal.ID, - Status: retrievalmarket.DealStatusAccepted, + Status: legacyretrievaltypes.DealStatusAccepted, } // Decide whether to accept the deal err := rv.acceptDeal(receiver, proposal, legacyProtocol, baseCid, selector) if err != nil { - response.Status = retrievalmarket.DealStatusRejected + response.Status = legacyretrievaltypes.DealStatusRejected response.Message = err.Error() return response, err } @@ -89,7 +91,7 @@ func (rv *requestValidator) validatePull(receiver peer.ID, proposal *retrievalma return response, nil } -func (rv *requestValidator) acceptDeal(receiver peer.ID, proposal *retrievalmarket.DealProposal, legacyProtocol bool, baseCid cid.Cid, selector ipld.Node) error { +func (rv *requestValidator) acceptDeal(receiver peer.ID, proposal *legacyretrievaltypes.DealProposal, legacyProtocol bool, baseCid cid.Cid, selector ipld.Node) error { // Check the proposal CID matches if proposal.PayloadCID != baseCid { return errors.New("incorrect CID for this proposal") @@ -112,7 +114,7 @@ func (rv *requestValidator) acceptDeal(receiver peer.ID, proposal *retrievalmark // Check if the piece is unsealed _, isUnsealed, err := rv.getPiece(proposal.PayloadCID, proposal.PieceCID) if err != nil { - if err == retrievalmarket.ErrNotFound { + if errors.Is(err, legacyretrievaltypes.ErrNotFound) { return fmt.Errorf("there is no piece containing payload cid %s: %w", proposal.PayloadCID, err) } return err @@ -139,7 +141,7 @@ func (rv *requestValidator) acceptDeal(receiver peer.ID, proposal *retrievalmark // Check the deal filter if rv.DealDecider != nil { - state := retrievalmarket.ProviderDealState{ + state := legacyretrievaltypes.ProviderDealState{ DealProposal: *proposal, Receiver: receiver, LegacyProtocol: legacyProtocol, @@ -175,16 +177,16 @@ func (rv *requestValidator) getPiece(payloadCid cid.Cid, pieceCID *cid.Cid) (Pie return PieceInfo{}, false, fmt.Errorf("piece cid not found for payload cid %s", payloadCid.String()) } -func (rv *requestValidator) Subscribe(subscriber retrievalmarket.ProviderValidationSubscriber) retrievalmarket.Unsubscribe { - return retrievalmarket.Unsubscribe(rv.psub.Subscribe(subscriber)) +func (rv *requestValidator) Subscribe(subscriber ProviderValidationSubscriber) legacyretrievaltypes.Unsubscribe { + return legacyretrievaltypes.Unsubscribe(rv.psub.Subscribe(subscriber)) } func queryValidationDispatcher(evt pubsub.Event, subscriberFn pubsub.SubscriberFn) error { - e, ok := evt.(retrievalmarket.ProviderValidationEvent) + e, ok := evt.(legacyretrievaltypes.ProviderValidationEvent) if !ok { return errors.New("wrong type of event") } - cb, ok := subscriberFn.(retrievalmarket.ProviderValidationSubscriber) + cb, ok := subscriberFn.(ProviderValidationSubscriber) if !ok { return errors.New("wrong type of callback") } diff --git a/retrievalmarket/testutil/testutil.go b/retrievalmarket/testutil/testutil.go new file mode 100644 index 000000000..ea7577e86 --- /dev/null +++ b/retrievalmarket/testutil/testutil.go @@ -0,0 +1,99 @@ +package testutil + +import ( + "context" + "os" + "testing" + + "github.com/filecoin-project/boost-graphsync/storeutil" + dtnet "github.com/filecoin-project/boost/datatransfer/network" + "github.com/ipfs/boxo/blockservice" + bstore "github.com/ipfs/boxo/blockstore" + "github.com/ipfs/boxo/exchange/offline" + "github.com/ipfs/boxo/ipld/merkledag" + "github.com/ipfs/go-datastore" + "github.com/ipfs/go-datastore/namespace" + dss "github.com/ipfs/go-datastore/sync" + ipldformat "github.com/ipfs/go-ipld-format" + "github.com/ipld/go-ipld-prime" + "github.com/libp2p/go-libp2p/core/host" + mocknet "github.com/libp2p/go-libp2p/p2p/net/mock" + "github.com/stretchr/testify/require" +) + +type Libp2pTestData struct { + Ctx context.Context + Ds1 datastore.Batching + Ds2 datastore.Batching + Bs1 bstore.Blockstore + Bs2 bstore.Blockstore + DagService1 ipldformat.DAGService + DagService2 ipldformat.DAGService + DTNet1 dtnet.DataTransferNetwork + DTNet2 dtnet.DataTransferNetwork + DTStore1 datastore.Batching + DTStore2 datastore.Batching + DTTmpDir1 string + DTTmpDir2 string + LinkSystem1 ipld.LinkSystem + LinkSystem2 ipld.LinkSystem + Host1 host.Host + Host2 host.Host + OrigBytes []byte + + MockNet mocknet.Mocknet +} + +func NewLibp2pTestData(ctx context.Context, t *testing.T) *Libp2pTestData { + testData := &Libp2pTestData{} + testData.Ctx = ctx + + var err error + + testData.Ds1 = dss.MutexWrap(datastore.NewMapDatastore()) + testData.Ds2 = dss.MutexWrap(datastore.NewMapDatastore()) + + // make a bstore and dag service + testData.Bs1 = bstore.NewBlockstore(testData.Ds1) + testData.Bs2 = bstore.NewBlockstore(testData.Ds2) + + testData.DagService1 = merkledag.NewDAGService(blockservice.New(testData.Bs1, offline.Exchange(testData.Bs1))) + testData.DagService2 = merkledag.NewDAGService(blockservice.New(testData.Bs2, offline.Exchange(testData.Bs2))) + + // setup an IPLD link system for bstore 1 + testData.LinkSystem1 = storeutil.LinkSystemForBlockstore(testData.Bs1) + + // setup an IPLD link system for bstore 2 + testData.LinkSystem2 = storeutil.LinkSystemForBlockstore(testData.Bs2) + + mn := mocknet.New() + + // setup network + testData.Host1, err = mn.GenPeer() + require.NoError(t, err) + + testData.Host2, err = mn.GenPeer() + require.NoError(t, err) + + err = mn.LinkAll() + require.NoError(t, err) + + testData.DTNet1 = dtnet.NewFromLibp2pHost(testData.Host1) + testData.DTNet2 = dtnet.NewFromLibp2pHost(testData.Host2) + + testData.DTStore1 = namespace.Wrap(testData.Ds1, datastore.NewKey("DataTransfer1")) + testData.DTStore2 = namespace.Wrap(testData.Ds1, datastore.NewKey("DataTransfer2")) + + testData.DTTmpDir1, err = os.MkdirTemp("", "dt-tmp-1") + require.NoError(t, err) + testData.DTTmpDir2, err = os.MkdirTemp("", "dt-tmp-2") + require.NoError(t, err) + t.Cleanup(func() { + _ = os.RemoveAll(testData.DTTmpDir1) + _ = os.RemoveAll(testData.DTTmpDir2) + }) + + testData.MockNet = mn + + return testData +} diff --git a/retrievalmarket/types/legacyretrievaltypes/bindoptions.go b/retrievalmarket/types/legacyretrievaltypes/bindoptions.go new file mode 100644 index 000000000..e95b233b7 --- /dev/null +++ b/retrievalmarket/types/legacyretrievaltypes/bindoptions.go @@ -0,0 +1,164 @@ +package legacyretrievaltypes + +import ( + "bytes" + "fmt" + "io" + + "github.com/filecoin-project/go-address" + "github.com/filecoin-project/go-state-types/abi" + "github.com/filecoin-project/go-state-types/big" + "github.com/filecoin-project/go-state-types/crypto" + "github.com/ipld/go-ipld-prime/codec/dagcbor" + "github.com/ipld/go-ipld-prime/datamodel" + "github.com/ipld/go-ipld-prime/node/basicnode" + "github.com/ipld/go-ipld-prime/node/bindnode" + "github.com/ipld/go-ipld-prime/schema" + cbg "github.com/whyrusleeping/cbor-gen" +) + +// go type converter functions for bindnode for common Filecoin data types + +// CborGenCompatibleNodeBindnodeOption converts a CborGenCompatibleNode type to +// and from an Any field in a schema +var CborGenCompatibleNodeBindnodeOption = bindnode.TypedAnyConverter(&CborGenCompatibleNode{}, cborGenCompatibleNodeFromAny, cborGenCompatibleNodeToAny) + +// BigIntBindnodeOption converts a big.Int type to and from a Bytes field in a +// schema +var BigIntBindnodeOption = bindnode.TypedBytesConverter(&big.Int{}, bigIntFromBytes, bigIntToBytes) + +// TokenAmountBindnodeOption converts a filecoin abi.TokenAmount type to and +// from a Bytes field in a schema +var TokenAmountBindnodeOption = bindnode.TypedBytesConverter(&abi.TokenAmount{}, tokenAmountFromBytes, tokenAmountToBytes) + +// AddressBindnodeOption converts a filecoin Address type to and from a Bytes +// field in a schema +var AddressBindnodeOption = bindnode.TypedBytesConverter(&address.Address{}, addressFromBytes, addressToBytes) + +// SignatureBindnodeOption converts a filecoin Signature type to and from a +// Bytes field in a schema +var SignatureBindnodeOption = bindnode.TypedBytesConverter(&crypto.Signature{}, signatureFromBytes, signatureToBytes) + +// CborGenCompatibleNode is for cbor-gen / go-ipld-prime compatibility, to +// replace Deferred types that are used to represent datamodel.Nodes. +// This shouldn't be used as a pointer (nullable/optional) as it can consume +// "Null" tokens and therefore be a Null. Instead, use +// CborGenCompatibleNode#IsNull to check for null status. +type CborGenCompatibleNode struct { + Node datamodel.Node +} + +func (sn CborGenCompatibleNode) IsNull() bool { + return sn.Node == nil || sn.Node == datamodel.Null +} + +// UnmarshalCBOR is for cbor-gen compatibility +func (sn *CborGenCompatibleNode) UnmarshalCBOR(r io.Reader) error { + // use cbg.Deferred.UnmarshalCBOR to figure out how much to pull + def := cbg.Deferred{} + if err := def.UnmarshalCBOR(r); err != nil { + return err + } + // convert it to a Node + na := basicnode.Prototype.Any.NewBuilder() + if err := dagcbor.Decode(na, bytes.NewReader(def.Raw)); err != nil { + return err + } + sn.Node = na.Build() + return nil +} + +// MarshalCBOR is for cbor-gen compatibility +func (sn *CborGenCompatibleNode) MarshalCBOR(w io.Writer) error { + node := datamodel.Null + if sn != nil && sn.Node != nil { + node = sn.Node + if tn, ok := node.(schema.TypedNode); ok { + node = tn.Representation() + } + } + return dagcbor.Encode(node, w) +} + +func cborGenCompatibleNodeFromAny(node datamodel.Node) (interface{}, error) { + return &CborGenCompatibleNode{Node: node}, nil +} + +func cborGenCompatibleNodeToAny(iface interface{}) (datamodel.Node, error) { + sn, ok := iface.(*CborGenCompatibleNode) + if !ok { + return nil, fmt.Errorf("expected *CborGenCompatibleNode value") + } + if sn.Node == nil { + return datamodel.Null, nil + } + return sn.Node, nil +} + +func tokenAmountFromBytes(b []byte) (interface{}, error) { + return bigIntFromBytes(b) +} + +func bigIntFromBytes(b []byte) (interface{}, error) { + if len(b) == 0 { + return big.NewInt(0), nil + } + return big.FromBytes(b) +} + +func tokenAmountToBytes(iface interface{}) ([]byte, error) { + return bigIntToBytes(iface) +} + +func bigIntToBytes(iface interface{}) ([]byte, error) { + bi, ok := iface.(*big.Int) + if !ok { + return nil, fmt.Errorf("expected *big.Int value") + } + if bi == nil || bi.Int == nil { + *bi = big.Zero() + } + return bi.Bytes() +} + +func addressFromBytes(b []byte) (interface{}, error) { + return address.NewFromBytes(b) +} + +func addressToBytes(iface interface{}) ([]byte, error) { + addr, ok := iface.(*address.Address) + if !ok { + return nil, fmt.Errorf("expected *Address value") + } + return addr.Bytes(), nil +} + +// Signature is a byteprefix union +func signatureFromBytes(b []byte) (interface{}, error) { + if len(b) > crypto.SignatureMaxLength { + return nil, fmt.Errorf("string too long") + } + if len(b) == 0 { + return nil, fmt.Errorf("string empty") + } + var s crypto.Signature + switch crypto.SigType(b[0]) { + default: + return nil, fmt.Errorf("invalid signature type in cbor input: %d", b[0]) + case crypto.SigTypeSecp256k1: + s.Type = crypto.SigTypeSecp256k1 + case crypto.SigTypeBLS: + s.Type = crypto.SigTypeBLS + } + s.Data = b[1:] + return &s, nil +} + +func signatureToBytes(iface interface{}) ([]byte, error) { + s, ok := iface.(*crypto.Signature) + if !ok { + return nil, fmt.Errorf("expected *Signature value") + } + ba := append([]byte{byte(s.Type)}, s.Data...) + return ba, nil +} diff --git a/retrievalmarket/types/legacyretrievaltypes/dealstatus.go b/retrievalmarket/types/legacyretrievaltypes/dealstatus.go new file mode 100644 index 000000000..5070e2537 --- /dev/null +++ b/retrievalmarket/types/legacyretrievaltypes/dealstatus.go @@ -0,0 +1,186 @@ +package legacyretrievaltypes + +import "fmt" + +// DealStatus is the status of a retrieval deal returned by a provider +// in a DealResponse +type DealStatus uint64 + +const ( + // DealStatusNew is a deal that nothing has happened with yet + DealStatusNew DealStatus = iota + + // DealStatusUnsealing means the provider is unsealing data + DealStatusUnsealing + + // DealStatusUnsealed means the provider has finished unsealing data + DealStatusUnsealed + + // DealStatusWaitForAcceptance means we're waiting to hear back if the provider accepted our deal + DealStatusWaitForAcceptance + + // DealStatusPaymentChannelCreating is the status set while waiting for the + // payment channel creation to complete + DealStatusPaymentChannelCreating + + // DealStatusPaymentChannelAddingFunds is the status when we are waiting for funds + // to finish being sent to the payment channel + DealStatusPaymentChannelAddingFunds + + // DealStatusAccepted means a deal has been accepted by a provider + // and its is ready to proceed with retrieval + DealStatusAccepted + + // DealStatusFundsNeededUnseal means a deal has been accepted by a provider + // and payment is needed to unseal the data + DealStatusFundsNeededUnseal + + // DealStatusFailing indicates something went wrong during a retrieval, + // and we are cleaning up before terminating with an error + DealStatusFailing + + // DealStatusRejected indicates the provider rejected a client's deal proposal + // for some reason + DealStatusRejected + + // DealStatusFundsNeeded indicates the provider needs a payment voucher to + // continue processing the deal + DealStatusFundsNeeded + + // DealStatusSendFunds indicates the client is now going to send funds because we reached the threshold of the last payment + DealStatusSendFunds + + // DealStatusSendFundsLastPayment indicates the client is now going to send final funds because + // we reached the threshold of the final payment + DealStatusSendFundsLastPayment + + // DealStatusOngoing indicates the provider is continuing to process a deal + DealStatusOngoing + + // DealStatusFundsNeededLastPayment indicates the provider needs a payment voucher + // in order to complete a deal + DealStatusFundsNeededLastPayment + + // DealStatusCompleted indicates a deal is complete + DealStatusCompleted + + // DealStatusDealNotFound indicates an update was received for a deal that could + // not be identified + DealStatusDealNotFound + + // DealStatusErrored indicates a deal has terminated in an error + DealStatusErrored + + // DealStatusBlocksComplete indicates that all blocks have been processed for the piece + DealStatusBlocksComplete + + // DealStatusFinalizing means the last payment has been received and + // we are just confirming the deal is complete + DealStatusFinalizing + + // DealStatusCompleting is just an inbetween state to perform final cleanup of + // complete deals + DealStatusCompleting + + // DealStatusCheckComplete is used for when the provided completes without a last payment + // requested cycle, to verify we have received all blocks + DealStatusCheckComplete + + // DealStatusCheckFunds means we are looking at the state of funding for the channel to determine + // if more money is incoming + DealStatusCheckFunds + + // DealStatusInsufficientFunds indicates we have depleted funds for the retrieval payment channel + // - we can resume after funds are added + DealStatusInsufficientFunds + + // DealStatusPaymentChannelAllocatingLane is the status when we are making a lane for this channel + DealStatusPaymentChannelAllocatingLane + + // DealStatusCancelling means we are cancelling an inprogress deal + DealStatusCancelling + + // DealStatusCancelled means a deal has been cancelled + DealStatusCancelled + + // DealStatusRetryLegacy means we're attempting the deal proposal for a second time using the legacy datatype + DealStatusRetryLegacy + + // DealStatusWaitForAcceptanceLegacy means we're waiting to hear the results on the legacy protocol + DealStatusWaitForAcceptanceLegacy + + // DealStatusClientWaitingForLastBlocks means that the provider has told + // the client that all blocks were sent for the deal, and the client is + // waiting for the last blocks to arrive. This should only happen when + // the deal price per byte is zero (if it's not zero the provider asks + // for final payment after sending the last blocks). + DealStatusClientWaitingForLastBlocks + + // DealStatusPaymentChannelAddingInitialFunds means that a payment channel + // exists from an earlier deal between client and provider, but we need + // to add funds to the channel for this particular deal + DealStatusPaymentChannelAddingInitialFunds + + // DealStatusErroring means that there was an error and we need to + // do some cleanup before moving to the error state + DealStatusErroring + + // DealStatusRejecting means that the deal was rejected and we need to do + // some cleanup before moving to the rejected state + DealStatusRejecting + + // DealStatusDealNotFoundCleanup means that the deal was not found and we + // need to do some cleanup before moving to the not found state + DealStatusDealNotFoundCleanup + + // DealStatusFinalizingBlockstore means that all blocks have been received, + // and the blockstore is being finalized + DealStatusFinalizingBlockstore +) + +// DealStatuses maps deal status to a human readable representation +var DealStatuses = map[DealStatus]string{ + DealStatusNew: "DealStatusNew", + DealStatusUnsealing: "DealStatusUnsealing", + DealStatusUnsealed: "DealStatusUnsealed", + DealStatusWaitForAcceptance: "DealStatusWaitForAcceptance", + DealStatusPaymentChannelCreating: "DealStatusPaymentChannelCreating", + DealStatusPaymentChannelAddingFunds: "DealStatusPaymentChannelAddingFunds", + DealStatusAccepted: "DealStatusAccepted", + DealStatusFundsNeededUnseal: "DealStatusFundsNeededUnseal", + DealStatusFailing: "DealStatusFailing", + DealStatusRejected: "DealStatusRejected", + DealStatusFundsNeeded: "DealStatusFundsNeeded", + DealStatusSendFunds: "DealStatusSendFunds", + DealStatusSendFundsLastPayment: "DealStatusSendFundsLastPayment", + DealStatusOngoing: "DealStatusOngoing", + DealStatusFundsNeededLastPayment: "DealStatusFundsNeededLastPayment", + DealStatusCompleted: "DealStatusCompleted", + DealStatusDealNotFound: "DealStatusDealNotFound", + DealStatusErrored: "DealStatusErrored", + DealStatusBlocksComplete: "DealStatusBlocksComplete", + DealStatusFinalizing: "DealStatusFinalizing", + DealStatusCompleting: "DealStatusCompleting", + DealStatusCheckComplete: "DealStatusCheckComplete", + DealStatusCheckFunds: "DealStatusCheckFunds", + DealStatusInsufficientFunds: "DealStatusInsufficientFunds", + DealStatusPaymentChannelAllocatingLane: "DealStatusPaymentChannelAllocatingLane", + DealStatusCancelling: "DealStatusCancelling", + DealStatusCancelled: "DealStatusCancelled", + DealStatusRetryLegacy: "DealStatusRetryLegacy", + DealStatusWaitForAcceptanceLegacy: "DealStatusWaitForAcceptanceLegacy", + DealStatusClientWaitingForLastBlocks: "DealStatusWaitingForLastBlocks", + DealStatusPaymentChannelAddingInitialFunds: "DealStatusPaymentChannelAddingInitialFunds", + DealStatusErroring: "DealStatusErroring", + DealStatusRejecting: "DealStatusRejecting", + DealStatusDealNotFoundCleanup: "DealStatusDealNotFoundCleanup", + DealStatusFinalizingBlockstore: "DealStatusFinalizingBlockstore", +} + +func (s DealStatus) String() string { + str, ok := DealStatuses[s] + if ok { + return str + } + return fmt.Sprintf("DealStatusUnknown - %d", s) +} diff --git a/retrievalmarket/types/legacyretrievaltypes/events.go b/retrievalmarket/types/legacyretrievaltypes/events.go new file mode 100644 index 000000000..a1a86079e --- /dev/null +++ b/retrievalmarket/types/legacyretrievaltypes/events.go @@ -0,0 +1,281 @@ +package legacyretrievaltypes + +import "fmt" + +// ClientEvent is an event that occurs in a deal lifecycle on the client +type ClientEvent uint64 + +const ( + // ClientEventOpen indicates a deal was initiated + ClientEventOpen ClientEvent = iota + + // ClientEventWriteDealProposalErrored means a network error writing a deal proposal + ClientEventWriteDealProposalErrored + + // ClientEventDealProposed means a deal was successfully sent to a miner + ClientEventDealProposed + + // ClientEventDealRejected means a deal was rejected by the provider + ClientEventDealRejected + + // ClientEventDealNotFound means a provider could not find a piece for a deal + ClientEventDealNotFound + + // ClientEventDealAccepted means a provider accepted a deal + ClientEventDealAccepted + + // ClientEventProviderCancelled means a provider has sent a message to cancel a deal + ClientEventProviderCancelled + + // ClientEventUnknownResponseReceived means a client received a response it doesn't + // understand from the provider + ClientEventUnknownResponseReceived + + // ClientEventPaymentChannelErrored means there was a failure creating a payment channel + ClientEventPaymentChannelErrored + + // ClientEventAllocateLaneErrored means there was a failure creating a lane in a payment channel + ClientEventAllocateLaneErrored + + // ClientEventPaymentChannelCreateInitiated means we are waiting for a message to + // create a payment channel to appear on chain + ClientEventPaymentChannelCreateInitiated + + // ClientEventPaymentChannelReady means the newly created payment channel is ready for the + // deal to resume + ClientEventPaymentChannelReady + + // ClientEventPaymentChannelAddingFunds mean we are waiting for funds to be + // added to a payment channel + ClientEventPaymentChannelAddingFunds + + // ClientEventPaymentChannelAddFundsErrored means that adding funds to the payment channel + // failed + ClientEventPaymentChannelAddFundsErrored + + // ClientEventLastPaymentRequested indicates the provider requested a final payment + ClientEventLastPaymentRequested + + // ClientEventAllBlocksReceived indicates the provider has sent all blocks + ClientEventAllBlocksReceived + + // ClientEventPaymentRequested indicates the provider requested a payment + ClientEventPaymentRequested + + // ClientEventUnsealPaymentRequested indicates the provider requested a payment for unsealing the sector + ClientEventUnsealPaymentRequested + + // ClientEventBlocksReceived indicates the provider has sent blocks + ClientEventBlocksReceived + + // ClientEventSendFunds emits when we reach the threshold to send the next payment + ClientEventSendFunds + + // ClientEventFundsExpended indicates a deal has run out of funds in the payment channel + // forcing the client to add more funds to continue the deal + ClientEventFundsExpended // when totalFunds is expended + + // ClientEventBadPaymentRequested indicates the provider asked for funds + // in a way that does not match the terms of the deal + ClientEventBadPaymentRequested + + // ClientEventCreateVoucherFailed indicates an error happened creating a payment voucher + ClientEventCreateVoucherFailed + + // ClientEventWriteDealPaymentErrored indicates a network error trying to write a payment + ClientEventWriteDealPaymentErrored + + // ClientEventPaymentSent indicates a payment was sent to the provider + ClientEventPaymentSent + + // ClientEventComplete is fired when the provider sends a message + // indicating that a deal has completed + ClientEventComplete + + // ClientEventDataTransferError emits when something go wrong at the data transfer level + ClientEventDataTransferError + + // ClientEventCancelComplete happens when a deal cancellation is transmitted to the provider + ClientEventCancelComplete + + // ClientEventEarlyTermination indications a provider send a deal complete without sending all data + ClientEventEarlyTermination + + // ClientEventCompleteVerified means that a provider completed without requesting a final payment but + // we verified we received all data + ClientEventCompleteVerified + + // ClientEventLaneAllocated is called when a lane is allocated + ClientEventLaneAllocated + + // ClientEventVoucherShortfall means we tried to create a voucher but did not have enough funds in channel + // to create it + ClientEventVoucherShortfall + + // ClientEventRecheckFunds runs when an external caller indicates there may be new funds in a payment channel + ClientEventRecheckFunds + + // ClientEventCancel runs when a user cancels a deal + ClientEventCancel + + // ClientEventWaitForLastBlocks is fired when the provider has told + // the client that all blocks were sent for the deal, and the client is + // waiting for the last blocks to arrive + ClientEventWaitForLastBlocks + + // ClientEventPaymentChannelSkip is fired when the total deal price is zero + // so there's no need to set up a payment channel + ClientEventPaymentChannelSkip + + // ClientEventPaymentNotSent indicates that payment was requested, but no + // payment was actually due, so a voucher was not sent to the provider + ClientEventPaymentNotSent + + // ClientEventBlockstoreFinalized is fired when the blockstore has been + // finalized after receiving all blocks + ClientEventBlockstoreFinalized + + // ClientEventFinalizeBlockstoreErrored is fired when there is an error + // finalizing the blockstore + ClientEventFinalizeBlockstoreErrored +) + +// ClientEvents is a human readable map of client event name -> event description +var ClientEvents = map[ClientEvent]string{ + ClientEventOpen: "ClientEventOpen", + ClientEventPaymentChannelErrored: "ClientEventPaymentChannelErrored", + ClientEventDealProposed: "ClientEventDealProposed", + ClientEventAllocateLaneErrored: "ClientEventAllocateLaneErrored", + ClientEventPaymentChannelCreateInitiated: "ClientEventPaymentChannelCreateInitiated", + ClientEventPaymentChannelReady: "ClientEventPaymentChannelReady", + ClientEventPaymentChannelAddingFunds: "ClientEventPaymentChannelAddingFunds", + ClientEventPaymentChannelAddFundsErrored: "ClientEventPaymentChannelAddFundsErrored", + ClientEventWriteDealProposalErrored: "ClientEventWriteDealProposalErrored", + ClientEventDealRejected: "ClientEventDealRejected", + ClientEventDealNotFound: "ClientEventDealNotFound", + ClientEventDealAccepted: "ClientEventDealAccepted", + ClientEventProviderCancelled: "ClientEventProviderCancelled", + ClientEventUnknownResponseReceived: "ClientEventUnknownResponseReceived", + ClientEventLastPaymentRequested: "ClientEventLastPaymentRequested", + ClientEventAllBlocksReceived: "ClientEventAllBlocksReceived", + ClientEventPaymentRequested: "ClientEventPaymentRequested", + ClientEventUnsealPaymentRequested: "ClientEventUnsealPaymentRequested", + ClientEventBlocksReceived: "ClientEventBlocksReceived", + ClientEventSendFunds: "ClientEventSendFunds", + ClientEventFundsExpended: "ClientEventFundsExpended", + ClientEventBadPaymentRequested: "ClientEventBadPaymentRequested", + ClientEventCreateVoucherFailed: "ClientEventCreateVoucherFailed", + ClientEventWriteDealPaymentErrored: "ClientEventWriteDealPaymentErrored", + ClientEventPaymentSent: "ClientEventPaymentSent", + ClientEventDataTransferError: "ClientEventDataTransferError", + ClientEventComplete: "ClientEventComplete", + ClientEventCancelComplete: "ClientEventCancelComplete", + ClientEventEarlyTermination: "ClientEventEarlyTermination", + ClientEventCompleteVerified: "ClientEventCompleteVerified", + ClientEventLaneAllocated: "ClientEventLaneAllocated", + ClientEventVoucherShortfall: "ClientEventVoucherShortfall", + ClientEventRecheckFunds: "ClientEventRecheckFunds", + ClientEventCancel: "ClientEventCancel", + ClientEventWaitForLastBlocks: "ClientEventWaitForLastBlocks", + ClientEventPaymentChannelSkip: "ClientEventPaymentChannelSkip", + ClientEventPaymentNotSent: "ClientEventPaymentNotSent", + ClientEventBlockstoreFinalized: "ClientEventBlockstoreFinalized", + ClientEventFinalizeBlockstoreErrored: "ClientEventFinalizeBlockstoreErrored", +} + +func (e ClientEvent) String() string { + s, ok := ClientEvents[e] + if ok { + return s + } + return fmt.Sprintf("ClientEventUnknown: %d", e) +} + +// ProviderEvent is an event that occurs in a deal lifecycle on the provider +type ProviderEvent uint64 + +const ( + // ProviderEventOpen indicates a new deal was received from a client + ProviderEventOpen ProviderEvent = iota + + // ProviderEventDealNotFound happens when the provider cannot find the piece for the + // deal proposed by the client + ProviderEventDealNotFound + + // ProviderEventDealRejected happens when a provider rejects a deal proposed + // by the client + ProviderEventDealRejected + + // ProviderEventDealAccepted happens when a provider accepts a deal + ProviderEventDealAccepted + + // ProviderEventBlockSent happens when the provider reads another block + // in the piece + ProviderEventBlockSent + + // ProviderEventBlocksCompleted happens when the provider reads the last block + // in the piece + ProviderEventBlocksCompleted + + // ProviderEventPaymentRequested happens when a provider asks for payment from + // a client for blocks sent + ProviderEventPaymentRequested + + // ProviderEventSaveVoucherFailed happens when an attempt to save a payment + // voucher fails + ProviderEventSaveVoucherFailed + + // ProviderEventPartialPaymentReceived happens when a provider receives and processes + // a payment that is less than what was requested to proceed with the deal + ProviderEventPartialPaymentReceived + + // ProviderEventPaymentReceived happens when a provider receives a payment + // and resumes processing a deal + ProviderEventPaymentReceived + + // ProviderEventComplete indicates a retrieval deal was completed for a client + ProviderEventComplete + + // ProviderEventUnsealError emits when something wrong occurs while unsealing data + ProviderEventUnsealError + + // ProviderEventUnsealComplete emits when the unsealing process is done + ProviderEventUnsealComplete + + // ProviderEventDataTransferError emits when something go wrong at the data transfer level + ProviderEventDataTransferError + + // ProviderEventCancelComplete happens when a deal cancellation is transmitted to the provider + ProviderEventCancelComplete + + // ProviderEventCleanupComplete happens when a deal is finished cleaning up and enters a complete state + ProviderEventCleanupComplete + + // ProviderEventMultiStoreError occurs when an error happens attempting to operate on the multistore + ProviderEventMultiStoreError + + // ProviderEventClientCancelled happens when the provider gets a cancel message from the client's data transfer + ProviderEventClientCancelled +) + +// ProviderEvents is a human readable map of provider event name -> event description +var ProviderEvents = map[ProviderEvent]string{ + ProviderEventOpen: "ProviderEventOpen", + ProviderEventDealNotFound: "ProviderEventDealNotFound", + ProviderEventDealRejected: "ProviderEventDealRejected", + ProviderEventDealAccepted: "ProviderEventDealAccepted", + ProviderEventBlockSent: "ProviderEventBlockSent", + ProviderEventBlocksCompleted: "ProviderEventBlocksCompleted", + ProviderEventPaymentRequested: "ProviderEventPaymentRequested", + ProviderEventSaveVoucherFailed: "ProviderEventSaveVoucherFailed", + ProviderEventPartialPaymentReceived: "ProviderEventPartialPaymentReceived", + ProviderEventPaymentReceived: "ProviderEventPaymentReceived", + ProviderEventComplete: "ProviderEventComplete", + ProviderEventUnsealError: "ProviderEventUnsealError", + ProviderEventUnsealComplete: "ProviderEventUnsealComplete", + ProviderEventDataTransferError: "ProviderEventDataTransferError", + ProviderEventCancelComplete: "ProviderEventCancelComplete", + ProviderEventCleanupComplete: "ProviderEventCleanupComplete", + ProviderEventMultiStoreError: "ProviderEventMultiStoreError", + ProviderEventClientCancelled: "ProviderEventClientCancelled", +} diff --git a/retrievalmarket/types/legacyretrievaltypes/migrations/maptypes/maptypes.go b/retrievalmarket/types/legacyretrievaltypes/migrations/maptypes/maptypes.go new file mode 100644 index 000000000..95908ef71 --- /dev/null +++ b/retrievalmarket/types/legacyretrievaltypes/migrations/maptypes/maptypes.go @@ -0,0 +1,55 @@ +package maptypes + +import ( + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/markets/piecestore" + "github.com/ipfs/go-cid" + "github.com/libp2p/go-libp2p/core/peer" + + "github.com/filecoin-project/go-address" + "github.com/filecoin-project/go-state-types/abi" + + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" +) + +//go:generate cbor-gen-for --map-encoding ClientDealState1 ProviderDealState1 + +// Version 1 of the ClientDealState +type ClientDealState1 struct { + legacyretrievaltypes.DealProposal + StoreID *uint64 + ChannelID datatransfer.ChannelID + LastPaymentRequested bool + AllBlocksReceived bool + TotalFunds abi.TokenAmount + ClientWallet address.Address + MinerWallet address.Address + PaymentInfo *legacyretrievaltypes.PaymentInfo + Status legacyretrievaltypes.DealStatus + Sender peer.ID + TotalReceived uint64 + Message string + BytesPaidFor uint64 + CurrentInterval uint64 + PaymentRequested abi.TokenAmount + FundsSpent abi.TokenAmount + UnsealFundsPaid abi.TokenAmount + WaitMsgCID *cid.Cid + VoucherShortfall abi.TokenAmount + LegacyProtocol bool +} + +// Version 1 of the ProviderDealState +type ProviderDealState1 struct { + legacyretrievaltypes.DealProposal + StoreID uint64 + ChannelID datatransfer.ChannelID + PieceInfo *piecestore.PieceInfo + Status legacyretrievaltypes.DealStatus + Receiver peer.ID + TotalSent uint64 + FundsReceived abi.TokenAmount + Message string + CurrentInterval uint64 + LegacyProtocol bool +} diff --git a/retrievalmarket/types/legacyretrievaltypes/migrations/maptypes/maptypes_cbor_gen.go b/retrievalmarket/types/legacyretrievaltypes/migrations/maptypes/maptypes_cbor_gen.go new file mode 100644 index 000000000..ff738788c --- /dev/null +++ b/retrievalmarket/types/legacyretrievaltypes/migrations/maptypes/maptypes_cbor_gen.go @@ -0,0 +1,1142 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package maptypes + +import ( + "fmt" + "io" + "math" + "sort" + + "github.com/filecoin-project/boost/markets/piecestore" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" + "github.com/ipfs/go-cid" + "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +func (t *ClientDealState1) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{181}); err != nil { + return err + } + + // t.Sender (peer.ID) (string) + if len("Sender") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Sender\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Sender"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Sender")); err != nil { + return err + } + + if len(t.Sender) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Sender was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Sender))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Sender)); err != nil { + return err + } + + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + if len("Status") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Status\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Status"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Status")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Status)); err != nil { + return err + } + + // t.Message (string) (string) + if len("Message") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Message\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Message"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Message")); err != nil { + return err + } + + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.StoreID (uint64) (uint64) + if len("StoreID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"StoreID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("StoreID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("StoreID")); err != nil { + return err + } + + if t.StoreID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(*t.StoreID)); err != nil { + return err + } + } + + // t.ChannelID (datatransfer.ChannelID) (struct) + if len("ChannelID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ChannelID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ChannelID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ChannelID")); err != nil { + return err + } + + if err := t.ChannelID.MarshalCBOR(cw); err != nil { + return err + } + + // t.FundsSpent (big.Int) (struct) + if len("FundsSpent") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"FundsSpent\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("FundsSpent"))); err != nil { + return err + } + if _, err := cw.WriteString(string("FundsSpent")); err != nil { + return err + } + + if err := t.FundsSpent.MarshalCBOR(cw); err != nil { + return err + } + + // t.TotalFunds (big.Int) (struct) + if len("TotalFunds") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"TotalFunds\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("TotalFunds"))); err != nil { + return err + } + if _, err := cw.WriteString(string("TotalFunds")); err != nil { + return err + } + + if err := t.TotalFunds.MarshalCBOR(cw); err != nil { + return err + } + + // t.WaitMsgCID (cid.Cid) (struct) + if len("WaitMsgCID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"WaitMsgCID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("WaitMsgCID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("WaitMsgCID")); err != nil { + return err + } + + if t.WaitMsgCID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.WaitMsgCID); err != nil { + return xerrors.Errorf("failed to write cid field t.WaitMsgCID: %w", err) + } + } + + // t.MinerWallet (address.Address) (struct) + if len("MinerWallet") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"MinerWallet\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("MinerWallet"))); err != nil { + return err + } + if _, err := cw.WriteString(string("MinerWallet")); err != nil { + return err + } + + if err := t.MinerWallet.MarshalCBOR(cw); err != nil { + return err + } + + // t.PaymentInfo (legacyretrievaltypes.PaymentInfo) (struct) + if len("PaymentInfo") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PaymentInfo\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PaymentInfo"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PaymentInfo")); err != nil { + return err + } + + if err := t.PaymentInfo.MarshalCBOR(cw); err != nil { + return err + } + + // t.BytesPaidFor (uint64) (uint64) + if len("BytesPaidFor") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"BytesPaidFor\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("BytesPaidFor"))); err != nil { + return err + } + if _, err := cw.WriteString(string("BytesPaidFor")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.BytesPaidFor)); err != nil { + return err + } + + // t.ClientWallet (address.Address) (struct) + if len("ClientWallet") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ClientWallet\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ClientWallet"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ClientWallet")); err != nil { + return err + } + + if err := t.ClientWallet.MarshalCBOR(cw); err != nil { + return err + } + + // t.DealProposal (legacyretrievaltypes.DealProposal) (struct) + if len("DealProposal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DealProposal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DealProposal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DealProposal")); err != nil { + return err + } + + if err := t.DealProposal.MarshalCBOR(cw); err != nil { + return err + } + + // t.TotalReceived (uint64) (uint64) + if len("TotalReceived") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"TotalReceived\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("TotalReceived"))); err != nil { + return err + } + if _, err := cw.WriteString(string("TotalReceived")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.TotalReceived)); err != nil { + return err + } + + // t.LegacyProtocol (bool) (bool) + if len("LegacyProtocol") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"LegacyProtocol\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("LegacyProtocol"))); err != nil { + return err + } + if _, err := cw.WriteString(string("LegacyProtocol")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.LegacyProtocol); err != nil { + return err + } + + // t.CurrentInterval (uint64) (uint64) + if len("CurrentInterval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"CurrentInterval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("CurrentInterval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("CurrentInterval")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.CurrentInterval)); err != nil { + return err + } + + // t.UnsealFundsPaid (big.Int) (struct) + if len("UnsealFundsPaid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"UnsealFundsPaid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("UnsealFundsPaid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("UnsealFundsPaid")); err != nil { + return err + } + + if err := t.UnsealFundsPaid.MarshalCBOR(cw); err != nil { + return err + } + + // t.PaymentRequested (big.Int) (struct) + if len("PaymentRequested") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PaymentRequested\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PaymentRequested"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PaymentRequested")); err != nil { + return err + } + + if err := t.PaymentRequested.MarshalCBOR(cw); err != nil { + return err + } + + // t.VoucherShortfall (big.Int) (struct) + if len("VoucherShortfall") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"VoucherShortfall\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("VoucherShortfall"))); err != nil { + return err + } + if _, err := cw.WriteString(string("VoucherShortfall")); err != nil { + return err + } + + if err := t.VoucherShortfall.MarshalCBOR(cw); err != nil { + return err + } + + // t.AllBlocksReceived (bool) (bool) + if len("AllBlocksReceived") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"AllBlocksReceived\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("AllBlocksReceived"))); err != nil { + return err + } + if _, err := cw.WriteString(string("AllBlocksReceived")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.AllBlocksReceived); err != nil { + return err + } + + // t.LastPaymentRequested (bool) (bool) + if len("LastPaymentRequested") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"LastPaymentRequested\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("LastPaymentRequested"))); err != nil { + return err + } + if _, err := cw.WriteString(string("LastPaymentRequested")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.LastPaymentRequested); err != nil { + return err + } + return nil +} + +func (t *ClientDealState1) UnmarshalCBOR(r io.Reader) (err error) { + *t = ClientDealState1{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("ClientDealState1: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Sender (peer.ID) (string) + case "Sender": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Sender = peer.ID(sval) + } + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + case "Status": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Status = legacyretrievaltypes.DealStatus(extra) + + } + // t.Message (string) (string) + case "Message": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.StoreID (uint64) (uint64) + case "StoreID": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + typed := uint64(extra) + t.StoreID = &typed + } + + } + // t.ChannelID (datatransfer.ChannelID) (struct) + case "ChannelID": + + { + + if err := t.ChannelID.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ChannelID: %w", err) + } + + } + // t.FundsSpent (big.Int) (struct) + case "FundsSpent": + + { + + if err := t.FundsSpent.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.FundsSpent: %w", err) + } + + } + // t.TotalFunds (big.Int) (struct) + case "TotalFunds": + + { + + if err := t.TotalFunds.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.TotalFunds: %w", err) + } + + } + // t.WaitMsgCID (cid.Cid) (struct) + case "WaitMsgCID": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.WaitMsgCID: %w", err) + } + + t.WaitMsgCID = &c + } + + } + // t.MinerWallet (address.Address) (struct) + case "MinerWallet": + + { + + if err := t.MinerWallet.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.MinerWallet: %w", err) + } + + } + // t.PaymentInfo (legacyretrievaltypes.PaymentInfo) (struct) + case "PaymentInfo": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.PaymentInfo = new(legacyretrievaltypes.PaymentInfo) + if err := t.PaymentInfo.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentInfo pointer: %w", err) + } + } + + } + // t.BytesPaidFor (uint64) (uint64) + case "BytesPaidFor": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.BytesPaidFor = uint64(extra) + + } + // t.ClientWallet (address.Address) (struct) + case "ClientWallet": + + { + + if err := t.ClientWallet.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ClientWallet: %w", err) + } + + } + // t.DealProposal (legacyretrievaltypes.DealProposal) (struct) + case "DealProposal": + + { + + if err := t.DealProposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DealProposal: %w", err) + } + + } + // t.TotalReceived (uint64) (uint64) + case "TotalReceived": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.TotalReceived = uint64(extra) + + } + // t.LegacyProtocol (bool) (bool) + case "LegacyProtocol": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.LegacyProtocol = false + case 21: + t.LegacyProtocol = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.CurrentInterval (uint64) (uint64) + case "CurrentInterval": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.CurrentInterval = uint64(extra) + + } + // t.UnsealFundsPaid (big.Int) (struct) + case "UnsealFundsPaid": + + { + + if err := t.UnsealFundsPaid.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.UnsealFundsPaid: %w", err) + } + + } + // t.PaymentRequested (big.Int) (struct) + case "PaymentRequested": + + { + + if err := t.PaymentRequested.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentRequested: %w", err) + } + + } + // t.VoucherShortfall (big.Int) (struct) + case "VoucherShortfall": + + { + + if err := t.VoucherShortfall.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.VoucherShortfall: %w", err) + } + + } + // t.AllBlocksReceived (bool) (bool) + case "AllBlocksReceived": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.AllBlocksReceived = false + case 21: + t.AllBlocksReceived = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.LastPaymentRequested (bool) (bool) + case "LastPaymentRequested": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.LastPaymentRequested = false + case 21: + t.LastPaymentRequested = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *ProviderDealState1) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{171}); err != nil { + return err + } + + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + if len("Status") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Status\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Status"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Status")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Status)); err != nil { + return err + } + + // t.Message (string) (string) + if len("Message") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Message\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Message"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Message")); err != nil { + return err + } + + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.StoreID (uint64) (uint64) + if len("StoreID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"StoreID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("StoreID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("StoreID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.StoreID)); err != nil { + return err + } + + // t.Receiver (peer.ID) (string) + if len("Receiver") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Receiver\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Receiver"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Receiver")); err != nil { + return err + } + + if len(t.Receiver) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Receiver was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Receiver))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Receiver)); err != nil { + return err + } + + // t.ChannelID (datatransfer.ChannelID) (struct) + if len("ChannelID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ChannelID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ChannelID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ChannelID")); err != nil { + return err + } + + if err := t.ChannelID.MarshalCBOR(cw); err != nil { + return err + } + + // t.PieceInfo (piecestore.PieceInfo) (struct) + if len("PieceInfo") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PieceInfo\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PieceInfo"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PieceInfo")); err != nil { + return err + } + + if err := t.PieceInfo.MarshalCBOR(cw); err != nil { + return err + } + + // t.TotalSent (uint64) (uint64) + if len("TotalSent") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"TotalSent\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("TotalSent"))); err != nil { + return err + } + if _, err := cw.WriteString(string("TotalSent")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.TotalSent)); err != nil { + return err + } + + // t.DealProposal (legacyretrievaltypes.DealProposal) (struct) + if len("DealProposal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DealProposal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DealProposal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DealProposal")); err != nil { + return err + } + + if err := t.DealProposal.MarshalCBOR(cw); err != nil { + return err + } + + // t.FundsReceived (big.Int) (struct) + if len("FundsReceived") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"FundsReceived\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("FundsReceived"))); err != nil { + return err + } + if _, err := cw.WriteString(string("FundsReceived")); err != nil { + return err + } + + if err := t.FundsReceived.MarshalCBOR(cw); err != nil { + return err + } + + // t.LegacyProtocol (bool) (bool) + if len("LegacyProtocol") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"LegacyProtocol\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("LegacyProtocol"))); err != nil { + return err + } + if _, err := cw.WriteString(string("LegacyProtocol")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.LegacyProtocol); err != nil { + return err + } + + // t.CurrentInterval (uint64) (uint64) + if len("CurrentInterval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"CurrentInterval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("CurrentInterval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("CurrentInterval")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.CurrentInterval)); err != nil { + return err + } + + return nil +} + +func (t *ProviderDealState1) UnmarshalCBOR(r io.Reader) (err error) { + *t = ProviderDealState1{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("ProviderDealState1: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + case "Status": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Status = legacyretrievaltypes.DealStatus(extra) + + } + // t.Message (string) (string) + case "Message": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.StoreID (uint64) (uint64) + case "StoreID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.StoreID = uint64(extra) + + } + // t.Receiver (peer.ID) (string) + case "Receiver": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Receiver = peer.ID(sval) + } + // t.ChannelID (datatransfer.ChannelID) (struct) + case "ChannelID": + + { + + if err := t.ChannelID.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ChannelID: %w", err) + } + + } + // t.PieceInfo (piecestore.PieceInfo) (struct) + case "PieceInfo": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.PieceInfo = new(piecestore.PieceInfo) + if err := t.PieceInfo.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PieceInfo pointer: %w", err) + } + } + + } + // t.TotalSent (uint64) (uint64) + case "TotalSent": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.TotalSent = uint64(extra) + + } + // t.DealProposal (legacyretrievaltypes.DealProposal) (struct) + case "DealProposal": + + { + + if err := t.DealProposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DealProposal: %w", err) + } + + } + // t.FundsReceived (big.Int) (struct) + case "FundsReceived": + + { + + if err := t.FundsReceived.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.FundsReceived: %w", err) + } + + } + // t.LegacyProtocol (bool) (bool) + case "LegacyProtocol": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.LegacyProtocol = false + case 21: + t.LegacyProtocol = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.CurrentInterval (uint64) (uint64) + case "CurrentInterval": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.CurrentInterval = uint64(extra) + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} diff --git a/retrievalmarket/types/legacyretrievaltypes/migrations/migrations.go b/retrievalmarket/types/legacyretrievaltypes/migrations/migrations.go new file mode 100644 index 000000000..85bfdb9e2 --- /dev/null +++ b/retrievalmarket/types/legacyretrievaltypes/migrations/migrations.go @@ -0,0 +1,386 @@ +package migrations + +import ( + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/markets/piecestore" + piecemigrations "github.com/filecoin-project/boost/markets/piecestore/migrations" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" + "github.com/ipfs/go-cid" + "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + + "github.com/filecoin-project/go-address" + versioning "github.com/filecoin-project/go-ds-versioning/pkg" + "github.com/filecoin-project/go-ds-versioning/pkg/versioned" + "github.com/filecoin-project/go-state-types/abi" + paychtypes "github.com/filecoin-project/go-state-types/builtin/v8/paych" + + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes/migrations/maptypes" +) + +//go:generate cbor-gen-for Query0 QueryResponse0 DealProposal0 DealResponse0 Params0 QueryParams0 DealPayment0 ClientDealState0 ProviderDealState0 PaymentInfo0 RetrievalPeer0 Ask0 + +// PaymentInfo0 is version 0 of PaymentInfo +type PaymentInfo0 struct { + PayCh address.Address + Lane uint64 +} + +// ClientDealState0 is version 0 of ClientDealState +type ClientDealState0 struct { + DealProposal0 + StoreID *uint64 + ChannelID datatransfer.ChannelID + LastPaymentRequested bool + AllBlocksReceived bool + TotalFunds abi.TokenAmount + ClientWallet address.Address + MinerWallet address.Address + PaymentInfo *PaymentInfo0 + Status legacyretrievaltypes.DealStatus + Sender peer.ID + TotalReceived uint64 + Message string + BytesPaidFor uint64 + CurrentInterval uint64 + PaymentRequested abi.TokenAmount + FundsSpent abi.TokenAmount + UnsealFundsPaid abi.TokenAmount + WaitMsgCID *cid.Cid // the CID of any message the client deal is waiting for + VoucherShortfall abi.TokenAmount +} + +// ProviderDealState0 is version 0 of ProviderDealState +type ProviderDealState0 struct { + DealProposal0 + StoreID uint64 + ChannelID datatransfer.ChannelID + PieceInfo *piecemigrations.PieceInfo0 + Status legacyretrievaltypes.DealStatus + Receiver peer.ID + TotalSent uint64 + FundsReceived abi.TokenAmount + Message string + CurrentInterval uint64 +} + +// RetrievalPeer0 is version 0 of RetrievalPeer +type RetrievalPeer0 struct { + Address address.Address + ID peer.ID // optional + PieceCID *cid.Cid +} + +// QueryParams0 is version 0 of QueryParams +type QueryParams0 struct { + PieceCID *cid.Cid // optional, query if miner has this cid in this piece. some miners may not be able to respond. + //Selector ipld.Node // optional, query if miner has this cid in this piece. some miners may not be able to respond. + //MaxPricePerByte abi.TokenAmount // optional, tell miner uninterested if more expensive than this + //MinPaymentInterval uint64 // optional, tell miner uninterested unless payment interval is greater than this + //MinPaymentIntervalIncrease uint64 // optional, tell miner uninterested unless payment interval increase is greater than this +} + +// Query0 is version 0 of Query +type Query0 struct { + PayloadCID cid.Cid // V0 + QueryParams0 // V1 +} + +// QueryResponse0 is version 0 of QueryResponse +type QueryResponse0 struct { + Status legacyretrievaltypes.QueryResponseStatus + PieceCIDFound legacyretrievaltypes.QueryItemStatus // V1 - if a PieceCID was requested, the result + //SelectorFound QueryItemStatus // V1 - if a Selector was requested, the result + + Size uint64 // Total size of piece in bytes + //ExpectedPayloadSize uint64 // V1 - optional, if PayloadCID + selector are specified and miner knows, can offer an expected size + + PaymentAddress address.Address // address to send funds to -- may be different than miner addr + MinPricePerByte abi.TokenAmount + MaxPaymentInterval uint64 + MaxPaymentIntervalIncrease uint64 + Message string + UnsealPrice abi.TokenAmount +} + +// Params0 is version 0 of Params +type Params0 struct { + Selector *cbg.Deferred // V1 + PieceCID *cid.Cid + PricePerByte abi.TokenAmount + PaymentInterval uint64 // when to request payment + PaymentIntervalIncrease uint64 + UnsealPrice abi.TokenAmount +} + +// DealProposal0 is version 0 of DealProposal +type DealProposal0 struct { + PayloadCID cid.Cid + ID legacyretrievaltypes.DealID + Params0 +} + +// Type method makes DealProposal0 usable as a voucher +func (dp *DealProposal0) Type() datatransfer.TypeIdentifier { + return "RetrievalDealProposal" +} + +// DealResponse0 is version 0 of DealResponse +type DealResponse0 struct { + Status legacyretrievaltypes.DealStatus + ID legacyretrievaltypes.DealID + + // payment required to proceed + PaymentOwed abi.TokenAmount + + Message string +} + +// Type method makes DealResponse0 usable as a voucher result +func (dr *DealResponse0) Type() datatransfer.TypeIdentifier { + return "RetrievalDealResponse" +} + +// DealPayment0 is version 0 of DealPayment +type DealPayment0 struct { + ID legacyretrievaltypes.DealID + PaymentChannel address.Address + PaymentVoucher *paychtypes.SignedVoucher +} + +// Type method makes DealPayment0 usable as a voucher +func (dr *DealPayment0) Type() datatransfer.TypeIdentifier { + return "RetrievalDealPayment" +} + +// Ask0 is version 0 of Ask +type Ask0 struct { + PricePerByte abi.TokenAmount + UnsealPrice abi.TokenAmount + PaymentInterval uint64 + PaymentIntervalIncrease uint64 +} + +// MigrateQueryParams0To1 migrates tuple encoded query params to map encoded query params +func MigrateQueryParams0To1(oldParams QueryParams0) legacyretrievaltypes.QueryParams { + return legacyretrievaltypes.QueryParams{ + PieceCID: oldParams.PieceCID, + } +} + +// MigrateQuery0To1 migrates tuple encoded query to map encoded query +func MigrateQuery0To1(oldQuery Query0) legacyretrievaltypes.Query { + return legacyretrievaltypes.Query{ + PayloadCID: oldQuery.PayloadCID, + QueryParams: MigrateQueryParams0To1(oldQuery.QueryParams0), + } +} + +// MigrateQueryResponse0To1 migrates tuple encoded query response to map encoded query response +func MigrateQueryResponse0To1(oldQr QueryResponse0) legacyretrievaltypes.QueryResponse { + return legacyretrievaltypes.QueryResponse{ + Status: oldQr.Status, + PieceCIDFound: oldQr.PieceCIDFound, + Size: oldQr.Size, + PaymentAddress: oldQr.PaymentAddress, + MinPricePerByte: oldQr.MinPricePerByte, + MaxPaymentInterval: oldQr.MaxPaymentInterval, + MaxPaymentIntervalIncrease: oldQr.MaxPaymentIntervalIncrease, + Message: oldQr.Message, + UnsealPrice: oldQr.UnsealPrice, + } +} + +// MigrateParams0To1 migrates tuple encoded deal params to map encoded deal params +func MigrateParams0To1(oldParams Params0) legacyretrievaltypes.Params { + return legacyretrievaltypes.Params{ + Selector: oldParams.Selector, + PieceCID: oldParams.PieceCID, + PricePerByte: oldParams.PricePerByte, + PaymentInterval: oldParams.PaymentInterval, + PaymentIntervalIncrease: oldParams.PaymentIntervalIncrease, + UnsealPrice: oldParams.UnsealPrice, + } +} + +// MigrateDealPayment0To1 migrates a tuple encoded DealPayment to a map +// encoded deal payment +func MigrateDealPayment0To1(oldDp DealPayment0) legacyretrievaltypes.DealPayment { + return legacyretrievaltypes.DealPayment{ + ID: oldDp.ID, + PaymentChannel: oldDp.PaymentChannel, + PaymentVoucher: oldDp.PaymentVoucher, + } +} + +// MigrateDealProposal0To1 migrates a tuple encoded DealProposal to a map +// encoded deal proposal +func MigrateDealProposal0To1(oldDp DealProposal0) legacyretrievaltypes.DealProposal { + return legacyretrievaltypes.DealProposal{ + PayloadCID: oldDp.PayloadCID, + ID: oldDp.ID, + Params: MigrateParams0To1(oldDp.Params0), + } +} + +// MigrateDealResponse0To1 migrates a tuple encoded DealResponse to a map +// encoded deal response +func MigrateDealResponse0To1(oldDr DealResponse0) legacyretrievaltypes.DealResponse { + return legacyretrievaltypes.DealResponse{ + Status: oldDr.Status, + ID: oldDr.ID, + PaymentOwed: oldDr.PaymentOwed, + Message: oldDr.Message, + } +} + +// MigratePaymentInfo0To1 migrates an optional payment info tuple encoded struct +// to a map encoded struct +func MigratePaymentInfo0To1(oldPi *PaymentInfo0) *legacyretrievaltypes.PaymentInfo { + if oldPi == nil { + return nil + } + return &legacyretrievaltypes.PaymentInfo{ + PayCh: oldPi.PayCh, + Lane: oldPi.Lane, + } +} + +// MigrateClientDealState0To1 migrates a tuple encoded deal state to a map encoded deal state +func MigrateClientDealState0To1(oldDs *ClientDealState0) (*maptypes.ClientDealState1, error) { + return &maptypes.ClientDealState1{ + DealProposal: MigrateDealProposal0To1(oldDs.DealProposal0), + StoreID: oldDs.StoreID, + ChannelID: oldDs.ChannelID, + LastPaymentRequested: oldDs.LastPaymentRequested, + AllBlocksReceived: oldDs.AllBlocksReceived, + TotalFunds: oldDs.TotalFunds, + ClientWallet: oldDs.ClientWallet, + MinerWallet: oldDs.MinerWallet, + PaymentInfo: MigratePaymentInfo0To1(oldDs.PaymentInfo), + Status: oldDs.Status, + Sender: oldDs.Sender, + TotalReceived: oldDs.TotalReceived, + Message: oldDs.Message, + BytesPaidFor: oldDs.BytesPaidFor, + CurrentInterval: oldDs.CurrentInterval, + PaymentRequested: oldDs.PaymentRequested, + FundsSpent: oldDs.FundsSpent, + UnsealFundsPaid: oldDs.UnsealFundsPaid, + WaitMsgCID: oldDs.WaitMsgCID, + VoucherShortfall: oldDs.VoucherShortfall, + LegacyProtocol: true, + }, nil +} + +// MigrateClientDealState1To2 migrates from v1 to v2 of a ClientDealState. +// The difference is that in v2 the ChannelID is a pointer, because the +// ChannelID is not set until the data transfer has started, so it should +// initially be nil. +func MigrateClientDealState1To2(oldDs *maptypes.ClientDealState1) (*legacyretrievaltypes.ClientDealState, error) { + var chid *datatransfer.ChannelID + if oldDs.ChannelID.Initiator != "" && oldDs.ChannelID.Responder != "" { + chid = &oldDs.ChannelID + } + return &legacyretrievaltypes.ClientDealState{ + DealProposal: oldDs.DealProposal, + StoreID: oldDs.StoreID, + ChannelID: chid, + LastPaymentRequested: oldDs.LastPaymentRequested, + AllBlocksReceived: oldDs.AllBlocksReceived, + TotalFunds: oldDs.TotalFunds, + ClientWallet: oldDs.ClientWallet, + MinerWallet: oldDs.MinerWallet, + PaymentInfo: oldDs.PaymentInfo, + Status: oldDs.Status, + Sender: oldDs.Sender, + TotalReceived: oldDs.TotalReceived, + Message: oldDs.Message, + BytesPaidFor: oldDs.BytesPaidFor, + CurrentInterval: oldDs.CurrentInterval, + PaymentRequested: oldDs.PaymentRequested, + FundsSpent: oldDs.FundsSpent, + UnsealFundsPaid: oldDs.UnsealFundsPaid, + WaitMsgCID: oldDs.WaitMsgCID, + VoucherShortfall: oldDs.VoucherShortfall, + LegacyProtocol: true, + }, nil +} + +// MigrateProviderDealState0To1 migrates a tuple encoded deal state to a map encoded deal state +func MigrateProviderDealState0To1(oldDs *ProviderDealState0) (*maptypes.ProviderDealState1, error) { + var pieceInfo *piecestore.PieceInfo + var err error + if oldDs.PieceInfo != nil { + pieceInfo, err = piecemigrations.MigratePieceInfo0To1(oldDs.PieceInfo) + if err != nil { + return nil, err + } + } + return &maptypes.ProviderDealState1{ + DealProposal: MigrateDealProposal0To1(oldDs.DealProposal0), + StoreID: oldDs.StoreID, + ChannelID: oldDs.ChannelID, + PieceInfo: pieceInfo, + Status: oldDs.Status, + Receiver: oldDs.Receiver, + TotalSent: oldDs.TotalSent, + FundsReceived: oldDs.FundsReceived, + Message: oldDs.Message, + CurrentInterval: oldDs.CurrentInterval, + LegacyProtocol: true, + }, nil +} + +// MigrateProviderDealState0To1 migrates from v1 to v2 of a +// MigrateProviderDealState. +// The difference is that in v2 the ChannelID is a pointer, because the +// ChannelID is not set until the data transfer has started, so it should +// initially be nil. +func MigrateProviderDealState1To2(oldDs *maptypes.ProviderDealState1) (*legacyretrievaltypes.ProviderDealState, error) { + var chid *datatransfer.ChannelID + if oldDs.ChannelID.Initiator != "" && oldDs.ChannelID.Responder != "" { + chid = &oldDs.ChannelID + } + return &legacyretrievaltypes.ProviderDealState{ + DealProposal: oldDs.DealProposal, + StoreID: oldDs.StoreID, + ChannelID: chid, + PieceInfo: oldDs.PieceInfo, + Status: oldDs.Status, + Receiver: oldDs.Receiver, + TotalSent: oldDs.TotalSent, + FundsReceived: oldDs.FundsReceived, + Message: oldDs.Message, + CurrentInterval: oldDs.CurrentInterval, + LegacyProtocol: oldDs.LegacyProtocol, + }, nil +} + +// MigrateAsk0To1 migrates a tuple encoded deal state to a map encoded deal state +func MigrateAsk0To1(oldAsk *Ask0) (*legacyretrievaltypes.Ask, error) { + return &legacyretrievaltypes.Ask{ + PricePerByte: oldAsk.PricePerByte, + UnsealPrice: oldAsk.UnsealPrice, + PaymentInterval: oldAsk.PaymentInterval, + PaymentIntervalIncrease: oldAsk.PaymentIntervalIncrease, + }, nil +} + +// ClientMigrations are migrations for the client's store of retrieval deals +var ClientMigrations = versioned.BuilderList{ + versioned.NewVersionedBuilder(MigrateClientDealState0To1, "1"), + versioned.NewVersionedBuilder(MigrateClientDealState1To2, "2").OldVersion("1"), +} + +// ProviderMigrations are migrations for the providers's store of retrieval deals +var ProviderMigrations = versioned.BuilderList{ + versioned.NewVersionedBuilder(MigrateProviderDealState0To1, "1"). + FilterKeys([]string{"/retrieval-ask", "/retrieval-ask/latest", "/retrieval-ask/1/latest", "/retrieval-ask/versions/current"}), + versioned.NewVersionedBuilder(MigrateProviderDealState1To2, "2").OldVersion("1"), +} + +// AskMigrations are migrations for the providers's retrieval ask +var AskMigrations = versioned.BuilderList{ + versioned.NewVersionedBuilder(MigrateAsk0To1, versioning.VersionKey("1")), +} diff --git a/retrievalmarket/types/legacyretrievaltypes/migrations/migrations_cbor_gen.go b/retrievalmarket/types/legacyretrievaltypes/migrations/migrations_cbor_gen.go new file mode 100644 index 000000000..c31956d96 --- /dev/null +++ b/retrievalmarket/types/legacyretrievaltypes/migrations/migrations_cbor_gen.go @@ -0,0 +1,1815 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package migrations + +import ( + "fmt" + "io" + "math" + "sort" + + migrations "github.com/filecoin-project/boost/markets/piecestore/migrations" + legacyretrievaltypes "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" + paych "github.com/filecoin-project/go-state-types/builtin/v8/paych" + cid "github.com/ipfs/go-cid" + peer "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + xerrors "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +var lengthBufQuery0 = []byte{130} + +func (t *Query0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufQuery0); err != nil { + return err + } + + // t.PayloadCID (cid.Cid) (struct) + + if err := cbg.WriteCid(cw, t.PayloadCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PayloadCID: %w", err) + } + + // t.QueryParams0 (migrations.QueryParams0) (struct) + if err := t.QueryParams0.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *Query0) UnmarshalCBOR(r io.Reader) (err error) { + *t = Query0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 2 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.PayloadCID (cid.Cid) (struct) + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PayloadCID: %w", err) + } + + t.PayloadCID = c + + } + // t.QueryParams0 (migrations.QueryParams0) (struct) + + { + + if err := t.QueryParams0.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.QueryParams0: %w", err) + } + + } + return nil +} + +var lengthBufQueryResponse0 = []byte{137} + +func (t *QueryResponse0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufQueryResponse0); err != nil { + return err + } + + // t.Status (legacyretrievaltypes.QueryResponseStatus) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Status)); err != nil { + return err + } + + // t.PieceCIDFound (legacyretrievaltypes.QueryItemStatus) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PieceCIDFound)); err != nil { + return err + } + + // t.Size (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Size)); err != nil { + return err + } + + // t.PaymentAddress (address.Address) (struct) + if err := t.PaymentAddress.MarshalCBOR(cw); err != nil { + return err + } + + // t.MinPricePerByte (big.Int) (struct) + if err := t.MinPricePerByte.MarshalCBOR(cw); err != nil { + return err + } + + // t.MaxPaymentInterval (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.MaxPaymentInterval)); err != nil { + return err + } + + // t.MaxPaymentIntervalIncrease (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.MaxPaymentIntervalIncrease)); err != nil { + return err + } + + // t.Message (string) (string) + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.UnsealPrice (big.Int) (struct) + if err := t.UnsealPrice.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *QueryResponse0) UnmarshalCBOR(r io.Reader) (err error) { + *t = QueryResponse0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 9 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Status (legacyretrievaltypes.QueryResponseStatus) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Status = legacyretrievaltypes.QueryResponseStatus(extra) + + } + // t.PieceCIDFound (legacyretrievaltypes.QueryItemStatus) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PieceCIDFound = legacyretrievaltypes.QueryItemStatus(extra) + + } + // t.Size (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Size = uint64(extra) + + } + // t.PaymentAddress (address.Address) (struct) + + { + + if err := t.PaymentAddress.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentAddress: %w", err) + } + + } + // t.MinPricePerByte (big.Int) (struct) + + { + + if err := t.MinPricePerByte.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.MinPricePerByte: %w", err) + } + + } + // t.MaxPaymentInterval (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.MaxPaymentInterval = uint64(extra) + + } + // t.MaxPaymentIntervalIncrease (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.MaxPaymentIntervalIncrease = uint64(extra) + + } + // t.Message (string) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.UnsealPrice (big.Int) (struct) + + { + + if err := t.UnsealPrice.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.UnsealPrice: %w", err) + } + + } + return nil +} + +var lengthBufDealProposal0 = []byte{131} + +func (t *DealProposal0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufDealProposal0); err != nil { + return err + } + + // t.PayloadCID (cid.Cid) (struct) + + if err := cbg.WriteCid(cw, t.PayloadCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PayloadCID: %w", err) + } + + // t.ID (legacyretrievaltypes.DealID) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.ID)); err != nil { + return err + } + + // t.Params0 (migrations.Params0) (struct) + if err := t.Params0.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *DealProposal0) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealProposal0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 3 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.PayloadCID (cid.Cid) (struct) + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PayloadCID: %w", err) + } + + t.PayloadCID = c + + } + // t.ID (legacyretrievaltypes.DealID) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.ID = legacyretrievaltypes.DealID(extra) + + } + // t.Params0 (migrations.Params0) (struct) + + { + + if err := t.Params0.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Params0: %w", err) + } + + } + return nil +} + +var lengthBufDealResponse0 = []byte{132} + +func (t *DealResponse0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufDealResponse0); err != nil { + return err + } + + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Status)); err != nil { + return err + } + + // t.ID (legacyretrievaltypes.DealID) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.ID)); err != nil { + return err + } + + // t.PaymentOwed (big.Int) (struct) + if err := t.PaymentOwed.MarshalCBOR(cw); err != nil { + return err + } + + // t.Message (string) (string) + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + return nil +} + +func (t *DealResponse0) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealResponse0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 4 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Status = legacyretrievaltypes.DealStatus(extra) + + } + // t.ID (legacyretrievaltypes.DealID) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.ID = legacyretrievaltypes.DealID(extra) + + } + // t.PaymentOwed (big.Int) (struct) + + { + + if err := t.PaymentOwed.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentOwed: %w", err) + } + + } + // t.Message (string) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + return nil +} + +var lengthBufParams0 = []byte{134} + +func (t *Params0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufParams0); err != nil { + return err + } + + // t.Selector (typegen.Deferred) (struct) + if err := t.Selector.MarshalCBOR(cw); err != nil { + return err + } + + // t.PieceCID (cid.Cid) (struct) + + if t.PieceCID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PieceCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PieceCID: %w", err) + } + } + + // t.PricePerByte (big.Int) (struct) + if err := t.PricePerByte.MarshalCBOR(cw); err != nil { + return err + } + + // t.PaymentInterval (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PaymentInterval)); err != nil { + return err + } + + // t.PaymentIntervalIncrease (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PaymentIntervalIncrease)); err != nil { + return err + } + + // t.UnsealPrice (big.Int) (struct) + if err := t.UnsealPrice.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *Params0) UnmarshalCBOR(r io.Reader) (err error) { + *t = Params0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 6 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Selector (typegen.Deferred) (struct) + + { + + t.Selector = new(cbg.Deferred) + + if err := t.Selector.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("failed to read deferred field: %w", err) + } + } + // t.PieceCID (cid.Cid) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PieceCID: %w", err) + } + + t.PieceCID = &c + } + + } + // t.PricePerByte (big.Int) (struct) + + { + + if err := t.PricePerByte.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PricePerByte: %w", err) + } + + } + // t.PaymentInterval (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PaymentInterval = uint64(extra) + + } + // t.PaymentIntervalIncrease (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PaymentIntervalIncrease = uint64(extra) + + } + // t.UnsealPrice (big.Int) (struct) + + { + + if err := t.UnsealPrice.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.UnsealPrice: %w", err) + } + + } + return nil +} + +var lengthBufQueryParams0 = []byte{129} + +func (t *QueryParams0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufQueryParams0); err != nil { + return err + } + + // t.PieceCID (cid.Cid) (struct) + + if t.PieceCID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PieceCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PieceCID: %w", err) + } + } + + return nil +} + +func (t *QueryParams0) UnmarshalCBOR(r io.Reader) (err error) { + *t = QueryParams0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 1 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.PieceCID (cid.Cid) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PieceCID: %w", err) + } + + t.PieceCID = &c + } + + } + return nil +} + +var lengthBufDealPayment0 = []byte{131} + +func (t *DealPayment0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufDealPayment0); err != nil { + return err + } + + // t.ID (legacyretrievaltypes.DealID) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.ID)); err != nil { + return err + } + + // t.PaymentChannel (address.Address) (struct) + if err := t.PaymentChannel.MarshalCBOR(cw); err != nil { + return err + } + + // t.PaymentVoucher (paych.SignedVoucher) (struct) + if err := t.PaymentVoucher.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *DealPayment0) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealPayment0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 3 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.ID (legacyretrievaltypes.DealID) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.ID = legacyretrievaltypes.DealID(extra) + + } + // t.PaymentChannel (address.Address) (struct) + + { + + if err := t.PaymentChannel.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentChannel: %w", err) + } + + } + // t.PaymentVoucher (paych.SignedVoucher) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.PaymentVoucher = new(paych.SignedVoucher) + if err := t.PaymentVoucher.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentVoucher pointer: %w", err) + } + } + + } + return nil +} + +var lengthBufClientDealState0 = []byte{148} + +func (t *ClientDealState0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufClientDealState0); err != nil { + return err + } + + // t.DealProposal0 (migrations.DealProposal0) (struct) + if err := t.DealProposal0.MarshalCBOR(cw); err != nil { + return err + } + + // t.StoreID (uint64) (uint64) + + if t.StoreID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(*t.StoreID)); err != nil { + return err + } + } + + // t.ChannelID (datatransfer.ChannelID) (struct) + if err := t.ChannelID.MarshalCBOR(cw); err != nil { + return err + } + + // t.LastPaymentRequested (bool) (bool) + if err := cbg.WriteBool(w, t.LastPaymentRequested); err != nil { + return err + } + + // t.AllBlocksReceived (bool) (bool) + if err := cbg.WriteBool(w, t.AllBlocksReceived); err != nil { + return err + } + + // t.TotalFunds (big.Int) (struct) + if err := t.TotalFunds.MarshalCBOR(cw); err != nil { + return err + } + + // t.ClientWallet (address.Address) (struct) + if err := t.ClientWallet.MarshalCBOR(cw); err != nil { + return err + } + + // t.MinerWallet (address.Address) (struct) + if err := t.MinerWallet.MarshalCBOR(cw); err != nil { + return err + } + + // t.PaymentInfo (migrations.PaymentInfo0) (struct) + if err := t.PaymentInfo.MarshalCBOR(cw); err != nil { + return err + } + + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Status)); err != nil { + return err + } + + // t.Sender (peer.ID) (string) + if len(t.Sender) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Sender was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Sender))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Sender)); err != nil { + return err + } + + // t.TotalReceived (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.TotalReceived)); err != nil { + return err + } + + // t.Message (string) (string) + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.BytesPaidFor (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.BytesPaidFor)); err != nil { + return err + } + + // t.CurrentInterval (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.CurrentInterval)); err != nil { + return err + } + + // t.PaymentRequested (big.Int) (struct) + if err := t.PaymentRequested.MarshalCBOR(cw); err != nil { + return err + } + + // t.FundsSpent (big.Int) (struct) + if err := t.FundsSpent.MarshalCBOR(cw); err != nil { + return err + } + + // t.UnsealFundsPaid (big.Int) (struct) + if err := t.UnsealFundsPaid.MarshalCBOR(cw); err != nil { + return err + } + + // t.WaitMsgCID (cid.Cid) (struct) + + if t.WaitMsgCID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.WaitMsgCID); err != nil { + return xerrors.Errorf("failed to write cid field t.WaitMsgCID: %w", err) + } + } + + // t.VoucherShortfall (big.Int) (struct) + if err := t.VoucherShortfall.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *ClientDealState0) UnmarshalCBOR(r io.Reader) (err error) { + *t = ClientDealState0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 20 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.DealProposal0 (migrations.DealProposal0) (struct) + + { + + if err := t.DealProposal0.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DealProposal0: %w", err) + } + + } + // t.StoreID (uint64) (uint64) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + typed := uint64(extra) + t.StoreID = &typed + } + + } + // t.ChannelID (datatransfer.ChannelID) (struct) + + { + + if err := t.ChannelID.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ChannelID: %w", err) + } + + } + // t.LastPaymentRequested (bool) (bool) + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.LastPaymentRequested = false + case 21: + t.LastPaymentRequested = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.AllBlocksReceived (bool) (bool) + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.AllBlocksReceived = false + case 21: + t.AllBlocksReceived = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.TotalFunds (big.Int) (struct) + + { + + if err := t.TotalFunds.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.TotalFunds: %w", err) + } + + } + // t.ClientWallet (address.Address) (struct) + + { + + if err := t.ClientWallet.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ClientWallet: %w", err) + } + + } + // t.MinerWallet (address.Address) (struct) + + { + + if err := t.MinerWallet.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.MinerWallet: %w", err) + } + + } + // t.PaymentInfo (migrations.PaymentInfo0) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.PaymentInfo = new(PaymentInfo0) + if err := t.PaymentInfo.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentInfo pointer: %w", err) + } + } + + } + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Status = legacyretrievaltypes.DealStatus(extra) + + } + // t.Sender (peer.ID) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Sender = peer.ID(sval) + } + // t.TotalReceived (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.TotalReceived = uint64(extra) + + } + // t.Message (string) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.BytesPaidFor (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.BytesPaidFor = uint64(extra) + + } + // t.CurrentInterval (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.CurrentInterval = uint64(extra) + + } + // t.PaymentRequested (big.Int) (struct) + + { + + if err := t.PaymentRequested.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentRequested: %w", err) + } + + } + // t.FundsSpent (big.Int) (struct) + + { + + if err := t.FundsSpent.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.FundsSpent: %w", err) + } + + } + // t.UnsealFundsPaid (big.Int) (struct) + + { + + if err := t.UnsealFundsPaid.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.UnsealFundsPaid: %w", err) + } + + } + // t.WaitMsgCID (cid.Cid) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.WaitMsgCID: %w", err) + } + + t.WaitMsgCID = &c + } + + } + // t.VoucherShortfall (big.Int) (struct) + + { + + if err := t.VoucherShortfall.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.VoucherShortfall: %w", err) + } + + } + return nil +} + +var lengthBufProviderDealState0 = []byte{138} + +func (t *ProviderDealState0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufProviderDealState0); err != nil { + return err + } + + // t.DealProposal0 (migrations.DealProposal0) (struct) + if err := t.DealProposal0.MarshalCBOR(cw); err != nil { + return err + } + + // t.StoreID (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.StoreID)); err != nil { + return err + } + + // t.ChannelID (datatransfer.ChannelID) (struct) + if err := t.ChannelID.MarshalCBOR(cw); err != nil { + return err + } + + // t.PieceInfo (migrations.PieceInfo0) (struct) + if err := t.PieceInfo.MarshalCBOR(cw); err != nil { + return err + } + + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Status)); err != nil { + return err + } + + // t.Receiver (peer.ID) (string) + if len(t.Receiver) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Receiver was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Receiver))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Receiver)); err != nil { + return err + } + + // t.TotalSent (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.TotalSent)); err != nil { + return err + } + + // t.FundsReceived (big.Int) (struct) + if err := t.FundsReceived.MarshalCBOR(cw); err != nil { + return err + } + + // t.Message (string) (string) + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.CurrentInterval (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.CurrentInterval)); err != nil { + return err + } + + return nil +} + +func (t *ProviderDealState0) UnmarshalCBOR(r io.Reader) (err error) { + *t = ProviderDealState0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 10 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.DealProposal0 (migrations.DealProposal0) (struct) + + { + + if err := t.DealProposal0.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DealProposal0: %w", err) + } + + } + // t.StoreID (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.StoreID = uint64(extra) + + } + // t.ChannelID (datatransfer.ChannelID) (struct) + + { + + if err := t.ChannelID.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ChannelID: %w", err) + } + + } + // t.PieceInfo (migrations.PieceInfo0) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.PieceInfo = new(migrations.PieceInfo0) + if err := t.PieceInfo.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PieceInfo pointer: %w", err) + } + } + + } + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Status = legacyretrievaltypes.DealStatus(extra) + + } + // t.Receiver (peer.ID) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Receiver = peer.ID(sval) + } + // t.TotalSent (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.TotalSent = uint64(extra) + + } + // t.FundsReceived (big.Int) (struct) + + { + + if err := t.FundsReceived.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.FundsReceived: %w", err) + } + + } + // t.Message (string) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.CurrentInterval (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.CurrentInterval = uint64(extra) + + } + return nil +} + +var lengthBufPaymentInfo0 = []byte{130} + +func (t *PaymentInfo0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufPaymentInfo0); err != nil { + return err + } + + // t.PayCh (address.Address) (struct) + if err := t.PayCh.MarshalCBOR(cw); err != nil { + return err + } + + // t.Lane (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Lane)); err != nil { + return err + } + + return nil +} + +func (t *PaymentInfo0) UnmarshalCBOR(r io.Reader) (err error) { + *t = PaymentInfo0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 2 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.PayCh (address.Address) (struct) + + { + + if err := t.PayCh.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PayCh: %w", err) + } + + } + // t.Lane (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Lane = uint64(extra) + + } + return nil +} + +var lengthBufRetrievalPeer0 = []byte{131} + +func (t *RetrievalPeer0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufRetrievalPeer0); err != nil { + return err + } + + // t.Address (address.Address) (struct) + if err := t.Address.MarshalCBOR(cw); err != nil { + return err + } + + // t.ID (peer.ID) (string) + if len(t.ID) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.ID was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.ID))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.ID)); err != nil { + return err + } + + // t.PieceCID (cid.Cid) (struct) + + if t.PieceCID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PieceCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PieceCID: %w", err) + } + } + + return nil +} + +func (t *RetrievalPeer0) UnmarshalCBOR(r io.Reader) (err error) { + *t = RetrievalPeer0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 3 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Address (address.Address) (struct) + + { + + if err := t.Address.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Address: %w", err) + } + + } + // t.ID (peer.ID) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.ID = peer.ID(sval) + } + // t.PieceCID (cid.Cid) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PieceCID: %w", err) + } + + t.PieceCID = &c + } + + } + return nil +} + +var lengthBufAsk0 = []byte{132} + +func (t *Ask0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufAsk0); err != nil { + return err + } + + // t.PricePerByte (big.Int) (struct) + if err := t.PricePerByte.MarshalCBOR(cw); err != nil { + return err + } + + // t.UnsealPrice (big.Int) (struct) + if err := t.UnsealPrice.MarshalCBOR(cw); err != nil { + return err + } + + // t.PaymentInterval (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PaymentInterval)); err != nil { + return err + } + + // t.PaymentIntervalIncrease (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PaymentIntervalIncrease)); err != nil { + return err + } + + return nil +} + +func (t *Ask0) UnmarshalCBOR(r io.Reader) (err error) { + *t = Ask0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 4 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.PricePerByte (big.Int) (struct) + + { + + if err := t.PricePerByte.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PricePerByte: %w", err) + } + + } + // t.UnsealPrice (big.Int) (struct) + + { + + if err := t.UnsealPrice.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.UnsealPrice: %w", err) + } + + } + // t.PaymentInterval (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PaymentInterval = uint64(extra) + + } + // t.PaymentIntervalIncrease (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PaymentIntervalIncrease = uint64(extra) + + } + return nil +} diff --git a/retrievalmarket/types/legacyretrievaltypes/types.go b/retrievalmarket/types/legacyretrievaltypes/types.go new file mode 100644 index 000000000..51c834bf5 --- /dev/null +++ b/retrievalmarket/types/legacyretrievaltypes/types.go @@ -0,0 +1,536 @@ +package legacyretrievaltypes + +import ( + "bytes" + _ "embed" + "errors" + "fmt" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/markets/piecestore" + "github.com/filecoin-project/boost/markets/shared" + "github.com/ipfs/go-cid" + "github.com/ipld/go-ipld-prime" + "github.com/ipld/go-ipld-prime/codec/dagcbor" + "github.com/ipld/go-ipld-prime/datamodel" + bindnoderegistry "github.com/ipld/go-ipld-prime/node/bindnode/registry" + "github.com/libp2p/go-libp2p/core/peer" + "github.com/libp2p/go-libp2p/core/protocol" + cbg "github.com/whyrusleeping/cbor-gen" + "golang.org/x/net/context" + "golang.org/x/xerrors" + + "github.com/filecoin-project/go-address" + "github.com/filecoin-project/go-state-types/abi" + "github.com/filecoin-project/go-state-types/big" + paychtypes "github.com/filecoin-project/go-state-types/builtin/v8/paych" +) + +//go:generate cbor-gen-for --map-encoding Query QueryResponse DealProposal DealResponse Params QueryParams DealPayment ClientDealState ProviderDealState PaymentInfo RetrievalPeer Ask + +// QueryProtocolID is the protocol for querying information about retrieval +// deal parameters +const QueryProtocolID = protocol.ID("/fil/retrieval/qry/1.0.0") + +// OldQueryProtocolID is the old query protocol for tuple structs +const OldQueryProtocolID = protocol.ID("/fil/retrieval/qry/0.0.1") + +// Unsubscribe is a function that unsubscribes a subscriber for either the +// client or the provider +type Unsubscribe func() + +// PaymentInfo is the payment channel and lane for a deal, once it is setup +type PaymentInfo struct { + PayCh address.Address + Lane uint64 +} + +// ClientDealState is the current state of a deal from the point of view +// of a retrieval client +type ClientDealState struct { + DealProposal + StoreID *uint64 + // Set when the data transfer is started + ChannelID *datatransfer.ChannelID + LastPaymentRequested bool + AllBlocksReceived bool + TotalFunds abi.TokenAmount + ClientWallet address.Address + MinerWallet address.Address + PaymentInfo *PaymentInfo + Status DealStatus + Sender peer.ID + TotalReceived uint64 + Message string + BytesPaidFor uint64 + CurrentInterval uint64 + PaymentRequested abi.TokenAmount + FundsSpent abi.TokenAmount + UnsealFundsPaid abi.TokenAmount + WaitMsgCID *cid.Cid // the CID of any message the client deal is waiting for + VoucherShortfall abi.TokenAmount + LegacyProtocol bool +} + +func (deal *ClientDealState) NextInterval() uint64 { + return deal.Params.NextInterval(deal.CurrentInterval) +} + +type ProviderQueryEvent struct { + Response QueryResponse + Error error +} + +type ProviderValidationEvent struct { + IsRestart bool + Receiver peer.ID + Proposal *DealProposal + BaseCid cid.Cid + Selector ipld.Node + Response *DealResponse + Error error +} + +// ProviderDealState is the current state of a deal from the point of view +// of a retrieval provider +type ProviderDealState struct { + DealProposal + StoreID uint64 + + ChannelID *datatransfer.ChannelID + PieceInfo *piecestore.PieceInfo + Status DealStatus + Receiver peer.ID + TotalSent uint64 + FundsReceived abi.TokenAmount + Message string + CurrentInterval uint64 + LegacyProtocol bool +} + +func (deal *ProviderDealState) IntervalLowerBound() uint64 { + return deal.Params.IntervalLowerBound(deal.CurrentInterval) +} + +func (deal *ProviderDealState) NextInterval() uint64 { + return deal.Params.NextInterval(deal.CurrentInterval) +} + +// Identifier provides a unique id for this provider deal +func (pds ProviderDealState) Identifier() ProviderDealIdentifier { + return ProviderDealIdentifier{Receiver: pds.Receiver, DealID: pds.ID} +} + +// ProviderDealIdentifier is a value that uniquely identifies a deal +type ProviderDealIdentifier struct { + Receiver peer.ID + DealID DealID +} + +func (p ProviderDealIdentifier) String() string { + return fmt.Sprintf("%v/%v", p.Receiver, p.DealID) +} + +// RetrievalPeer is a provider address/peer.ID pair (everything needed to make +// deals for with a miner) +type RetrievalPeer struct { + Address address.Address + ID peer.ID // optional + PieceCID *cid.Cid +} + +// QueryResponseStatus indicates whether a queried piece is available +type QueryResponseStatus uint64 + +const ( + // QueryResponseAvailable indicates a provider has a piece and is prepared to + // return it + QueryResponseAvailable QueryResponseStatus = iota + + // QueryResponseUnavailable indicates a provider either does not have or cannot + // serve the queried piece to the client + QueryResponseUnavailable + + // QueryResponseError indicates something went wrong generating a query response + QueryResponseError +) + +// QueryItemStatus (V1) indicates whether the requested part of a piece (payload or selector) +// is available for retrieval +type QueryItemStatus uint64 + +const ( + // QueryItemAvailable indicates requested part of the piece is available to be + // served + QueryItemAvailable QueryItemStatus = iota + + // QueryItemUnavailable indicates the piece either does not contain the requested + // item or it cannot be served + QueryItemUnavailable + + // QueryItemUnknown indicates the provider cannot determine if the given item + // is part of the requested piece (for example, if the piece is sealed and the + // miner does not maintain a payload CID index) + QueryItemUnknown +) + +// QueryParams - V1 - indicate what specific information about a piece that a retrieval +// client is interested in, as well as specific parameters the client is seeking +// for the retrieval deal +type QueryParams struct { + PieceCID *cid.Cid // optional, query if miner has this cid in this piece. some miners may not be able to respond. + //Selector ipld.Node // optional, query if miner has this cid in this piece. some miners may not be able to respond. + //MaxPricePerByte abi.TokenAmount // optional, tell miner uninterested if more expensive than this + //MinPaymentInterval uint64 // optional, tell miner uninterested unless payment interval is greater than this + //MinPaymentIntervalIncrease uint64 // optional, tell miner uninterested unless payment interval increase is greater than this +} + +// Query is a query to a given provider to determine information about a piece +// they may have available for retrieval +type Query struct { + PayloadCID cid.Cid // V0 + QueryParams // V1 +} + +// QueryUndefined is a query with no values +var QueryUndefined = Query{} + +// NewQueryV0 creates a V0 query (which only specifies a payload) +func NewQueryV0(payloadCID cid.Cid) Query { + return Query{PayloadCID: payloadCID} +} + +// NewQueryV1 creates a V1 query (which has an optional pieceCID) +func NewQueryV1(payloadCID cid.Cid, pieceCID *cid.Cid) Query { + return Query{ + PayloadCID: payloadCID, + QueryParams: QueryParams{ + PieceCID: pieceCID, + }, + } +} + +// QueryResponse is a miners response to a given retrieval query +type QueryResponse struct { + Status QueryResponseStatus + PieceCIDFound QueryItemStatus // V1 - if a PieceCID was requested, the result + //SelectorFound QueryItemStatus // V1 - if a Selector was requested, the result + + Size uint64 // Total size of piece in bytes + //ExpectedPayloadSize uint64 // V1 - optional, if PayloadCID + selector are specified and miner knows, can offer an expected size + + PaymentAddress address.Address // address to send funds to -- may be different than miner addr + MinPricePerByte abi.TokenAmount + MaxPaymentInterval uint64 + MaxPaymentIntervalIncrease uint64 + Message string + UnsealPrice abi.TokenAmount +} + +// QueryResponseUndefined is an empty QueryResponse +var QueryResponseUndefined = QueryResponse{} + +// PieceRetrievalPrice is the total price to retrieve the piece (size * MinPricePerByte + UnsealedPrice) +func (qr QueryResponse) PieceRetrievalPrice() abi.TokenAmount { + return big.Add(big.Mul(qr.MinPricePerByte, abi.NewTokenAmount(int64(qr.Size))), qr.UnsealPrice) +} + +// PayloadRetrievalPrice is the expected price to retrieve just the given payload +// & selector (V1) +//func (qr QueryResponse) PayloadRetrievalPrice() abi.TokenAmount { +// return types.BigMul(qr.MinPricePerByte, types.NewInt(qr.ExpectedPayloadSize)) +//} + +// IsTerminalError returns true if this status indicates processing of this deal +// is complete with an error +func IsTerminalError(status DealStatus) bool { + return status == DealStatusDealNotFound || + status == DealStatusFailing || + status == DealStatusRejected +} + +// IsTerminalSuccess returns true if this status indicates processing of this deal +// is complete with a success +func IsTerminalSuccess(status DealStatus) bool { + return status == DealStatusCompleted +} + +// IsTerminalStatus returns true if this status indicates processing of a deal is +// complete (either success or error) +func IsTerminalStatus(status DealStatus) bool { + return IsTerminalError(status) || IsTerminalSuccess(status) +} + +// Params are the parameters requested for a retrieval deal proposal +type Params struct { + //Selector CborGenCompatibleNode // V1 + Selector *cbg.Deferred + PieceCID *cid.Cid + PricePerByte abi.TokenAmount + PaymentInterval uint64 // when to request payment + PaymentIntervalIncrease uint64 + UnsealPrice abi.TokenAmount +} + +func (p Params) SelectorSpecified() bool { + return p.Selector != nil && !bytes.Equal(p.Selector.Raw, cbg.CborNull) +} + +func (p Params) IntervalLowerBound(currentInterval uint64) uint64 { + intervalSize := p.PaymentInterval + var lowerBound uint64 + var target uint64 + for target < currentInterval { + lowerBound = target + target += intervalSize + intervalSize += p.PaymentIntervalIncrease + } + return lowerBound +} + +func (p Params) NextInterval(currentInterval uint64) uint64 { + intervalSize := p.PaymentInterval + var nextInterval uint64 + for nextInterval <= currentInterval { + nextInterval += intervalSize + intervalSize += p.PaymentIntervalIncrease + } + return nextInterval +} + +// NewParamsV0 generates parameters for a retrieval deal, which is always a whole piece deal +func NewParamsV0(pricePerByte abi.TokenAmount, paymentInterval uint64, paymentIntervalIncrease uint64) Params { + return Params{ + PricePerByte: pricePerByte, + PaymentInterval: paymentInterval, + PaymentIntervalIncrease: paymentIntervalIncrease, + UnsealPrice: big.Zero(), + } +} + +// NewParamsV1 generates parameters for a retrieval deal, including a selector +func NewParamsV1(pricePerByte abi.TokenAmount, paymentInterval uint64, paymentIntervalIncrease uint64, sel ipld.Node, pieceCid *cid.Cid, unsealPrice abi.TokenAmount) (Params, error) { + var buffer bytes.Buffer + + if sel == nil { + return Params{}, xerrors.New("selector required for NewParamsV1") + } + + err := dagcbor.Encode(sel, &buffer) + if err != nil { + return Params{}, xerrors.Errorf("error encoding selector: %w", err) + } + + return Params{ + Selector: &cbg.Deferred{Raw: buffer.Bytes()}, + PieceCID: pieceCid, + PricePerByte: pricePerByte, + PaymentInterval: paymentInterval, + PaymentIntervalIncrease: paymentIntervalIncrease, + UnsealPrice: unsealPrice, + }, nil +} + +// DealID is an identifier for a retrieval deal (unique to a client) +type DealID uint64 + +func (d DealID) String() string { + return fmt.Sprintf("%d", d) +} + +// DealProposal is a proposal for a new retrieval deal +type DealProposal struct { + PayloadCID cid.Cid + ID DealID + Params +} + +// Type method makes DealProposal usable as a voucher +func (dp *DealProposal) Type() datatransfer.TypeIdentifier { + return "RetrievalDealProposal/1" +} + +// DealProposalUndefined is an undefined deal proposal +var DealProposalUndefined = DealProposal{} + +// DealProposalType is the DealProposal voucher type +const DealProposalType = datatransfer.TypeIdentifier("RetrievalDealProposal/1") + +func DealProposalFromNode(node datamodel.Node) (*DealProposal, error) { + if node == nil { + return nil, fmt.Errorf("empty voucher") + } + dpIface, err := BindnodeRegistry.TypeFromNode(node, &DealProposal{}) + if err != nil { + return nil, xerrors.Errorf("invalid DealProposal: %w", err) + } + dp, _ := dpIface.(*DealProposal) // safe to assume type + return dp, nil +} + +// DealResponse is a response to a retrieval deal proposal +type DealResponse struct { + Status DealStatus + ID DealID + + // payment required to proceed + PaymentOwed abi.TokenAmount + + Message string +} + +// Type method makes DealResponse usable as a voucher result +func (dr *DealResponse) Type() datatransfer.TypeIdentifier { + return "RetrievalDealResponse/1" +} + +// DealResponseUndefined is an undefined deal response +var DealResponseUndefined = DealResponse{} + +// DealPayment is a payment for an in progress retrieval deal +type DealPayment struct { + ID DealID + PaymentChannel address.Address + PaymentVoucher *paychtypes.SignedVoucher +} + +// Type method makes DealPayment usable as a voucher +func (dr *DealPayment) Type() datatransfer.TypeIdentifier { + return "RetrievalDealPayment/1" +} + +// DealPaymentUndefined is an undefined deal payment +var DealPaymentUndefined = DealPayment{} + +var ( + // ErrNotFound means a piece was not found during retrieval + ErrNotFound = errors.New("not found") + + // ErrVerification means a retrieval contained a block response that did not verify + ErrVerification = errors.New("Error when verify data") +) + +type Ask struct { + PricePerByte abi.TokenAmount + UnsealPrice abi.TokenAmount + PaymentInterval uint64 + PaymentIntervalIncrease uint64 +} + +// ShortfallErorr is an error that indicates a short fall of funds +type ShortfallError struct { + shortfall abi.TokenAmount +} + +// NewShortfallError returns a new error indicating a shortfall of funds +func NewShortfallError(shortfall abi.TokenAmount) error { + return ShortfallError{shortfall} +} + +// Shortfall returns the numerical value of the shortfall +func (se ShortfallError) Shortfall() abi.TokenAmount { + return se.shortfall +} +func (se ShortfallError) Error() string { + return fmt.Sprintf("Inssufficient Funds. Shortfall: %s", se.shortfall.String()) +} + +// ChannelAvailableFunds provides information about funds in a channel +type ChannelAvailableFunds struct { + // ConfirmedAmt is the amount of funds that have been confirmed on-chain + // for the channel + ConfirmedAmt abi.TokenAmount + // PendingAmt is the amount of funds that are pending confirmation on-chain + PendingAmt abi.TokenAmount + // PendingWaitSentinel can be used with PaychGetWaitReady to wait for + // confirmation of pending funds + PendingWaitSentinel *cid.Cid + // QueuedAmt is the amount that is queued up behind a pending request + QueuedAmt abi.TokenAmount + // VoucherRedeemedAmt is the amount that is redeemed by vouchers on-chain + // and in the local datastore + VoucherReedeemedAmt abi.TokenAmount +} + +// PricingInput provides input parameters required to price a retrieval deal. +type PricingInput struct { + // PayloadCID is the cid of the payload to retrieve. + PayloadCID cid.Cid + // PieceCID is the cid of the Piece from which the Payload will be retrieved. + PieceCID cid.Cid + // PieceSize is the size of the Piece from which the payload will be retrieved. + PieceSize abi.UnpaddedPieceSize + // Client is the peerID of the retrieval client. + Client peer.ID + // VerifiedDeal is true if there exists a verified storage deal for the PayloadCID. + VerifiedDeal bool + // Unsealed is true if there exists an unsealed sector from which we can retrieve the given payload. + Unsealed bool + // CurrentAsk is the current configured ask in the ask-store. + CurrentAsk Ask +} + +// RetrievalClient is a client interface for making retrieval deals +type RetrievalClient interface { + + // NextID generates a new deal ID. + NextID() DealID + + // Start initializes the client by running migrations + Start(ctx context.Context) error + + // OnReady registers a listener for when the client comes on line + OnReady(shared.ReadyFunc) + + // Find Providers finds retrieval providers who may be storing a given piece + FindProviders(payloadCID cid.Cid) []RetrievalPeer + + // Query asks a provider for information about a piece it is storing + Query( + ctx context.Context, + p RetrievalPeer, + payloadCID cid.Cid, + params QueryParams, + ) (QueryResponse, error) + + // Retrieve retrieves all or part of a piece with the given retrieval parameters + Retrieve( + ctx context.Context, + id DealID, + payloadCID cid.Cid, + params Params, + totalFunds abi.TokenAmount, + p RetrievalPeer, + clientWallet address.Address, + minerWallet address.Address, + ) (DealID, error) + + // SubscribeToEvents listens for events that happen related to client retrievals + SubscribeToEvents(subscriber ClientSubscriber) Unsubscribe + + // V1 + + // TryRestartInsufficientFunds attempts to restart any deals stuck in the insufficient funds state + // after funds are added to a given payment channel + TryRestartInsufficientFunds(paymentChannel address.Address) error + + // CancelDeal attempts to cancel an inprogress deal + CancelDeal(id DealID) error + + // GetDeal returns a given deal by deal ID, if it exists + GetDeal(dealID DealID) (ClientDealState, error) + + // ListDeals returns all deals + ListDeals() (map[DealID]ClientDealState, error) +} + +// ClientSubscriber is a callback that is registered to listen for retrieval events +type ClientSubscriber func(event ClientEvent, state ClientDealState) + +var BindnodeRegistry = bindnoderegistry.NewRegistry() + +// DealResponseType is the DealResponse usable as a voucher type +const DealResponseType = datatransfer.TypeIdentifier("RetrievalDealResponse/1") + +// DealPaymentType is the DealPayment voucher type +const DealPaymentType = datatransfer.TypeIdentifier("RetrievalDealPayment/1") diff --git a/retrievalmarket/types/legacyretrievaltypes/types_cbor_gen.go b/retrievalmarket/types/legacyretrievaltypes/types_cbor_gen.go new file mode 100644 index 000000000..1cfbd86be --- /dev/null +++ b/retrievalmarket/types/legacyretrievaltypes/types_cbor_gen.go @@ -0,0 +1,2909 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package legacyretrievaltypes + +import ( + "fmt" + "io" + "math" + "sort" + + datatransfer "github.com/filecoin-project/boost/datatransfer" + piecestore "github.com/filecoin-project/boost/markets/piecestore" + paych "github.com/filecoin-project/go-state-types/builtin/v8/paych" + cid "github.com/ipfs/go-cid" + peer "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + xerrors "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +func (t *Query) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.PayloadCID (cid.Cid) (struct) + if len("PayloadCID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PayloadCID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PayloadCID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PayloadCID")); err != nil { + return err + } + + if err := cbg.WriteCid(cw, t.PayloadCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PayloadCID: %w", err) + } + + // t.QueryParams (legacyretrievaltypes.QueryParams) (struct) + if len("QueryParams") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"QueryParams\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("QueryParams"))); err != nil { + return err + } + if _, err := cw.WriteString(string("QueryParams")); err != nil { + return err + } + + if err := t.QueryParams.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *Query) UnmarshalCBOR(r io.Reader) (err error) { + *t = Query{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("Query: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.PayloadCID (cid.Cid) (struct) + case "PayloadCID": + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PayloadCID: %w", err) + } + + t.PayloadCID = c + + } + // t.QueryParams (legacyretrievaltypes.QueryParams) (struct) + case "QueryParams": + + { + + if err := t.QueryParams.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.QueryParams: %w", err) + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *QueryResponse) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{169}); err != nil { + return err + } + + // t.Size (uint64) (uint64) + if len("Size") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Size\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Size"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Size")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Size)); err != nil { + return err + } + + // t.Status (legacyretrievaltypes.QueryResponseStatus) (uint64) + if len("Status") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Status\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Status"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Status")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Status)); err != nil { + return err + } + + // t.Message (string) (string) + if len("Message") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Message\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Message"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Message")); err != nil { + return err + } + + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.UnsealPrice (big.Int) (struct) + if len("UnsealPrice") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"UnsealPrice\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("UnsealPrice"))); err != nil { + return err + } + if _, err := cw.WriteString(string("UnsealPrice")); err != nil { + return err + } + + if err := t.UnsealPrice.MarshalCBOR(cw); err != nil { + return err + } + + // t.PieceCIDFound (legacyretrievaltypes.QueryItemStatus) (uint64) + if len("PieceCIDFound") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PieceCIDFound\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PieceCIDFound"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PieceCIDFound")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PieceCIDFound)); err != nil { + return err + } + + // t.PaymentAddress (address.Address) (struct) + if len("PaymentAddress") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PaymentAddress\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PaymentAddress"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PaymentAddress")); err != nil { + return err + } + + if err := t.PaymentAddress.MarshalCBOR(cw); err != nil { + return err + } + + // t.MinPricePerByte (big.Int) (struct) + if len("MinPricePerByte") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"MinPricePerByte\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("MinPricePerByte"))); err != nil { + return err + } + if _, err := cw.WriteString(string("MinPricePerByte")); err != nil { + return err + } + + if err := t.MinPricePerByte.MarshalCBOR(cw); err != nil { + return err + } + + // t.MaxPaymentInterval (uint64) (uint64) + if len("MaxPaymentInterval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"MaxPaymentInterval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("MaxPaymentInterval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("MaxPaymentInterval")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.MaxPaymentInterval)); err != nil { + return err + } + + // t.MaxPaymentIntervalIncrease (uint64) (uint64) + if len("MaxPaymentIntervalIncrease") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"MaxPaymentIntervalIncrease\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("MaxPaymentIntervalIncrease"))); err != nil { + return err + } + if _, err := cw.WriteString(string("MaxPaymentIntervalIncrease")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.MaxPaymentIntervalIncrease)); err != nil { + return err + } + + return nil +} + +func (t *QueryResponse) UnmarshalCBOR(r io.Reader) (err error) { + *t = QueryResponse{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("QueryResponse: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Size (uint64) (uint64) + case "Size": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Size = uint64(extra) + + } + // t.Status (legacyretrievaltypes.QueryResponseStatus) (uint64) + case "Status": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Status = QueryResponseStatus(extra) + + } + // t.Message (string) (string) + case "Message": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.UnsealPrice (big.Int) (struct) + case "UnsealPrice": + + { + + if err := t.UnsealPrice.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.UnsealPrice: %w", err) + } + + } + // t.PieceCIDFound (legacyretrievaltypes.QueryItemStatus) (uint64) + case "PieceCIDFound": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PieceCIDFound = QueryItemStatus(extra) + + } + // t.PaymentAddress (address.Address) (struct) + case "PaymentAddress": + + { + + if err := t.PaymentAddress.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentAddress: %w", err) + } + + } + // t.MinPricePerByte (big.Int) (struct) + case "MinPricePerByte": + + { + + if err := t.MinPricePerByte.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.MinPricePerByte: %w", err) + } + + } + // t.MaxPaymentInterval (uint64) (uint64) + case "MaxPaymentInterval": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.MaxPaymentInterval = uint64(extra) + + } + // t.MaxPaymentIntervalIncrease (uint64) (uint64) + case "MaxPaymentIntervalIncrease": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.MaxPaymentIntervalIncrease = uint64(extra) + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *DealProposal) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{163}); err != nil { + return err + } + + // t.ID (legacyretrievaltypes.DealID) (uint64) + if len("ID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.ID)); err != nil { + return err + } + + // t.Params (legacyretrievaltypes.Params) (struct) + if len("Params") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Params\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Params"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Params")); err != nil { + return err + } + + if err := t.Params.MarshalCBOR(cw); err != nil { + return err + } + + // t.PayloadCID (cid.Cid) (struct) + if len("PayloadCID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PayloadCID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PayloadCID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PayloadCID")); err != nil { + return err + } + + if err := cbg.WriteCid(cw, t.PayloadCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PayloadCID: %w", err) + } + + return nil +} + +func (t *DealProposal) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealProposal{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("DealProposal: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.ID (legacyretrievaltypes.DealID) (uint64) + case "ID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.ID = DealID(extra) + + } + // t.Params (legacyretrievaltypes.Params) (struct) + case "Params": + + { + + if err := t.Params.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Params: %w", err) + } + + } + // t.PayloadCID (cid.Cid) (struct) + case "PayloadCID": + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PayloadCID: %w", err) + } + + t.PayloadCID = c + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *DealResponse) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{164}); err != nil { + return err + } + + // t.ID (legacyretrievaltypes.DealID) (uint64) + if len("ID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.ID)); err != nil { + return err + } + + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + if len("Status") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Status\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Status"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Status")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Status)); err != nil { + return err + } + + // t.Message (string) (string) + if len("Message") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Message\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Message"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Message")); err != nil { + return err + } + + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.PaymentOwed (big.Int) (struct) + if len("PaymentOwed") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PaymentOwed\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PaymentOwed"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PaymentOwed")); err != nil { + return err + } + + if err := t.PaymentOwed.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *DealResponse) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealResponse{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("DealResponse: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.ID (legacyretrievaltypes.DealID) (uint64) + case "ID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.ID = DealID(extra) + + } + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + case "Status": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Status = DealStatus(extra) + + } + // t.Message (string) (string) + case "Message": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.PaymentOwed (big.Int) (struct) + case "PaymentOwed": + + { + + if err := t.PaymentOwed.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentOwed: %w", err) + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *Params) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{166}); err != nil { + return err + } + + // t.PieceCID (cid.Cid) (struct) + if len("PieceCID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PieceCID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PieceCID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PieceCID")); err != nil { + return err + } + + if t.PieceCID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PieceCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PieceCID: %w", err) + } + } + + // t.Selector (typegen.Deferred) (struct) + if len("Selector") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Selector\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Selector"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Selector")); err != nil { + return err + } + + if err := t.Selector.MarshalCBOR(cw); err != nil { + return err + } + + // t.UnsealPrice (big.Int) (struct) + if len("UnsealPrice") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"UnsealPrice\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("UnsealPrice"))); err != nil { + return err + } + if _, err := cw.WriteString(string("UnsealPrice")); err != nil { + return err + } + + if err := t.UnsealPrice.MarshalCBOR(cw); err != nil { + return err + } + + // t.PricePerByte (big.Int) (struct) + if len("PricePerByte") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PricePerByte\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PricePerByte"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PricePerByte")); err != nil { + return err + } + + if err := t.PricePerByte.MarshalCBOR(cw); err != nil { + return err + } + + // t.PaymentInterval (uint64) (uint64) + if len("PaymentInterval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PaymentInterval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PaymentInterval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PaymentInterval")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PaymentInterval)); err != nil { + return err + } + + // t.PaymentIntervalIncrease (uint64) (uint64) + if len("PaymentIntervalIncrease") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PaymentIntervalIncrease\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PaymentIntervalIncrease"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PaymentIntervalIncrease")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PaymentIntervalIncrease)); err != nil { + return err + } + + return nil +} + +func (t *Params) UnmarshalCBOR(r io.Reader) (err error) { + *t = Params{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("Params: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.PieceCID (cid.Cid) (struct) + case "PieceCID": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PieceCID: %w", err) + } + + t.PieceCID = &c + } + + } + // t.Selector (typegen.Deferred) (struct) + case "Selector": + + { + + t.Selector = new(cbg.Deferred) + + if err := t.Selector.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("failed to read deferred field: %w", err) + } + } + // t.UnsealPrice (big.Int) (struct) + case "UnsealPrice": + + { + + if err := t.UnsealPrice.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.UnsealPrice: %w", err) + } + + } + // t.PricePerByte (big.Int) (struct) + case "PricePerByte": + + { + + if err := t.PricePerByte.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PricePerByte: %w", err) + } + + } + // t.PaymentInterval (uint64) (uint64) + case "PaymentInterval": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PaymentInterval = uint64(extra) + + } + // t.PaymentIntervalIncrease (uint64) (uint64) + case "PaymentIntervalIncrease": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PaymentIntervalIncrease = uint64(extra) + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *QueryParams) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{161}); err != nil { + return err + } + + // t.PieceCID (cid.Cid) (struct) + if len("PieceCID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PieceCID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PieceCID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PieceCID")); err != nil { + return err + } + + if t.PieceCID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PieceCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PieceCID: %w", err) + } + } + + return nil +} + +func (t *QueryParams) UnmarshalCBOR(r io.Reader) (err error) { + *t = QueryParams{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("QueryParams: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.PieceCID (cid.Cid) (struct) + case "PieceCID": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PieceCID: %w", err) + } + + t.PieceCID = &c + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *DealPayment) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{163}); err != nil { + return err + } + + // t.ID (legacyretrievaltypes.DealID) (uint64) + if len("ID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.ID)); err != nil { + return err + } + + // t.PaymentChannel (address.Address) (struct) + if len("PaymentChannel") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PaymentChannel\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PaymentChannel"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PaymentChannel")); err != nil { + return err + } + + if err := t.PaymentChannel.MarshalCBOR(cw); err != nil { + return err + } + + // t.PaymentVoucher (paych.SignedVoucher) (struct) + if len("PaymentVoucher") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PaymentVoucher\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PaymentVoucher"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PaymentVoucher")); err != nil { + return err + } + + if err := t.PaymentVoucher.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *DealPayment) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealPayment{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("DealPayment: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.ID (legacyretrievaltypes.DealID) (uint64) + case "ID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.ID = DealID(extra) + + } + // t.PaymentChannel (address.Address) (struct) + case "PaymentChannel": + + { + + if err := t.PaymentChannel.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentChannel: %w", err) + } + + } + // t.PaymentVoucher (paych.SignedVoucher) (struct) + case "PaymentVoucher": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.PaymentVoucher = new(paych.SignedVoucher) + if err := t.PaymentVoucher.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentVoucher pointer: %w", err) + } + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *ClientDealState) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{181}); err != nil { + return err + } + + // t.Sender (peer.ID) (string) + if len("Sender") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Sender\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Sender"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Sender")); err != nil { + return err + } + + if len(t.Sender) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Sender was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Sender))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Sender)); err != nil { + return err + } + + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + if len("Status") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Status\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Status"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Status")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Status)); err != nil { + return err + } + + // t.Message (string) (string) + if len("Message") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Message\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Message"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Message")); err != nil { + return err + } + + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.StoreID (uint64) (uint64) + if len("StoreID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"StoreID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("StoreID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("StoreID")); err != nil { + return err + } + + if t.StoreID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(*t.StoreID)); err != nil { + return err + } + } + + // t.ChannelID (datatransfer.ChannelID) (struct) + if len("ChannelID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ChannelID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ChannelID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ChannelID")); err != nil { + return err + } + + if err := t.ChannelID.MarshalCBOR(cw); err != nil { + return err + } + + // t.FundsSpent (big.Int) (struct) + if len("FundsSpent") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"FundsSpent\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("FundsSpent"))); err != nil { + return err + } + if _, err := cw.WriteString(string("FundsSpent")); err != nil { + return err + } + + if err := t.FundsSpent.MarshalCBOR(cw); err != nil { + return err + } + + // t.TotalFunds (big.Int) (struct) + if len("TotalFunds") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"TotalFunds\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("TotalFunds"))); err != nil { + return err + } + if _, err := cw.WriteString(string("TotalFunds")); err != nil { + return err + } + + if err := t.TotalFunds.MarshalCBOR(cw); err != nil { + return err + } + + // t.WaitMsgCID (cid.Cid) (struct) + if len("WaitMsgCID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"WaitMsgCID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("WaitMsgCID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("WaitMsgCID")); err != nil { + return err + } + + if t.WaitMsgCID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.WaitMsgCID); err != nil { + return xerrors.Errorf("failed to write cid field t.WaitMsgCID: %w", err) + } + } + + // t.MinerWallet (address.Address) (struct) + if len("MinerWallet") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"MinerWallet\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("MinerWallet"))); err != nil { + return err + } + if _, err := cw.WriteString(string("MinerWallet")); err != nil { + return err + } + + if err := t.MinerWallet.MarshalCBOR(cw); err != nil { + return err + } + + // t.PaymentInfo (legacyretrievaltypes.PaymentInfo) (struct) + if len("PaymentInfo") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PaymentInfo\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PaymentInfo"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PaymentInfo")); err != nil { + return err + } + + if err := t.PaymentInfo.MarshalCBOR(cw); err != nil { + return err + } + + // t.BytesPaidFor (uint64) (uint64) + if len("BytesPaidFor") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"BytesPaidFor\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("BytesPaidFor"))); err != nil { + return err + } + if _, err := cw.WriteString(string("BytesPaidFor")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.BytesPaidFor)); err != nil { + return err + } + + // t.ClientWallet (address.Address) (struct) + if len("ClientWallet") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ClientWallet\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ClientWallet"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ClientWallet")); err != nil { + return err + } + + if err := t.ClientWallet.MarshalCBOR(cw); err != nil { + return err + } + + // t.DealProposal (legacyretrievaltypes.DealProposal) (struct) + if len("DealProposal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DealProposal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DealProposal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DealProposal")); err != nil { + return err + } + + if err := t.DealProposal.MarshalCBOR(cw); err != nil { + return err + } + + // t.TotalReceived (uint64) (uint64) + if len("TotalReceived") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"TotalReceived\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("TotalReceived"))); err != nil { + return err + } + if _, err := cw.WriteString(string("TotalReceived")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.TotalReceived)); err != nil { + return err + } + + // t.LegacyProtocol (bool) (bool) + if len("LegacyProtocol") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"LegacyProtocol\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("LegacyProtocol"))); err != nil { + return err + } + if _, err := cw.WriteString(string("LegacyProtocol")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.LegacyProtocol); err != nil { + return err + } + + // t.CurrentInterval (uint64) (uint64) + if len("CurrentInterval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"CurrentInterval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("CurrentInterval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("CurrentInterval")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.CurrentInterval)); err != nil { + return err + } + + // t.UnsealFundsPaid (big.Int) (struct) + if len("UnsealFundsPaid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"UnsealFundsPaid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("UnsealFundsPaid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("UnsealFundsPaid")); err != nil { + return err + } + + if err := t.UnsealFundsPaid.MarshalCBOR(cw); err != nil { + return err + } + + // t.PaymentRequested (big.Int) (struct) + if len("PaymentRequested") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PaymentRequested\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PaymentRequested"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PaymentRequested")); err != nil { + return err + } + + if err := t.PaymentRequested.MarshalCBOR(cw); err != nil { + return err + } + + // t.VoucherShortfall (big.Int) (struct) + if len("VoucherShortfall") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"VoucherShortfall\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("VoucherShortfall"))); err != nil { + return err + } + if _, err := cw.WriteString(string("VoucherShortfall")); err != nil { + return err + } + + if err := t.VoucherShortfall.MarshalCBOR(cw); err != nil { + return err + } + + // t.AllBlocksReceived (bool) (bool) + if len("AllBlocksReceived") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"AllBlocksReceived\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("AllBlocksReceived"))); err != nil { + return err + } + if _, err := cw.WriteString(string("AllBlocksReceived")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.AllBlocksReceived); err != nil { + return err + } + + // t.LastPaymentRequested (bool) (bool) + if len("LastPaymentRequested") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"LastPaymentRequested\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("LastPaymentRequested"))); err != nil { + return err + } + if _, err := cw.WriteString(string("LastPaymentRequested")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.LastPaymentRequested); err != nil { + return err + } + return nil +} + +func (t *ClientDealState) UnmarshalCBOR(r io.Reader) (err error) { + *t = ClientDealState{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("ClientDealState: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Sender (peer.ID) (string) + case "Sender": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Sender = peer.ID(sval) + } + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + case "Status": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Status = DealStatus(extra) + + } + // t.Message (string) (string) + case "Message": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.StoreID (uint64) (uint64) + case "StoreID": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + typed := uint64(extra) + t.StoreID = &typed + } + + } + // t.ChannelID (datatransfer.ChannelID) (struct) + case "ChannelID": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.ChannelID = new(datatransfer.ChannelID) + if err := t.ChannelID.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ChannelID pointer: %w", err) + } + } + + } + // t.FundsSpent (big.Int) (struct) + case "FundsSpent": + + { + + if err := t.FundsSpent.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.FundsSpent: %w", err) + } + + } + // t.TotalFunds (big.Int) (struct) + case "TotalFunds": + + { + + if err := t.TotalFunds.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.TotalFunds: %w", err) + } + + } + // t.WaitMsgCID (cid.Cid) (struct) + case "WaitMsgCID": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.WaitMsgCID: %w", err) + } + + t.WaitMsgCID = &c + } + + } + // t.MinerWallet (address.Address) (struct) + case "MinerWallet": + + { + + if err := t.MinerWallet.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.MinerWallet: %w", err) + } + + } + // t.PaymentInfo (legacyretrievaltypes.PaymentInfo) (struct) + case "PaymentInfo": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.PaymentInfo = new(PaymentInfo) + if err := t.PaymentInfo.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentInfo pointer: %w", err) + } + } + + } + // t.BytesPaidFor (uint64) (uint64) + case "BytesPaidFor": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.BytesPaidFor = uint64(extra) + + } + // t.ClientWallet (address.Address) (struct) + case "ClientWallet": + + { + + if err := t.ClientWallet.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ClientWallet: %w", err) + } + + } + // t.DealProposal (legacyretrievaltypes.DealProposal) (struct) + case "DealProposal": + + { + + if err := t.DealProposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DealProposal: %w", err) + } + + } + // t.TotalReceived (uint64) (uint64) + case "TotalReceived": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.TotalReceived = uint64(extra) + + } + // t.LegacyProtocol (bool) (bool) + case "LegacyProtocol": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.LegacyProtocol = false + case 21: + t.LegacyProtocol = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.CurrentInterval (uint64) (uint64) + case "CurrentInterval": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.CurrentInterval = uint64(extra) + + } + // t.UnsealFundsPaid (big.Int) (struct) + case "UnsealFundsPaid": + + { + + if err := t.UnsealFundsPaid.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.UnsealFundsPaid: %w", err) + } + + } + // t.PaymentRequested (big.Int) (struct) + case "PaymentRequested": + + { + + if err := t.PaymentRequested.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PaymentRequested: %w", err) + } + + } + // t.VoucherShortfall (big.Int) (struct) + case "VoucherShortfall": + + { + + if err := t.VoucherShortfall.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.VoucherShortfall: %w", err) + } + + } + // t.AllBlocksReceived (bool) (bool) + case "AllBlocksReceived": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.AllBlocksReceived = false + case 21: + t.AllBlocksReceived = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.LastPaymentRequested (bool) (bool) + case "LastPaymentRequested": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.LastPaymentRequested = false + case 21: + t.LastPaymentRequested = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *ProviderDealState) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{171}); err != nil { + return err + } + + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + if len("Status") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Status\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Status"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Status")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Status)); err != nil { + return err + } + + // t.Message (string) (string) + if len("Message") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Message\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Message"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Message")); err != nil { + return err + } + + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.StoreID (uint64) (uint64) + if len("StoreID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"StoreID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("StoreID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("StoreID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.StoreID)); err != nil { + return err + } + + // t.Receiver (peer.ID) (string) + if len("Receiver") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Receiver\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Receiver"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Receiver")); err != nil { + return err + } + + if len(t.Receiver) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Receiver was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Receiver))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Receiver)); err != nil { + return err + } + + // t.ChannelID (datatransfer.ChannelID) (struct) + if len("ChannelID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ChannelID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ChannelID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ChannelID")); err != nil { + return err + } + + if err := t.ChannelID.MarshalCBOR(cw); err != nil { + return err + } + + // t.PieceInfo (piecestore.PieceInfo) (struct) + if len("PieceInfo") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PieceInfo\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PieceInfo"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PieceInfo")); err != nil { + return err + } + + if err := t.PieceInfo.MarshalCBOR(cw); err != nil { + return err + } + + // t.TotalSent (uint64) (uint64) + if len("TotalSent") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"TotalSent\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("TotalSent"))); err != nil { + return err + } + if _, err := cw.WriteString(string("TotalSent")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.TotalSent)); err != nil { + return err + } + + // t.DealProposal (legacyretrievaltypes.DealProposal) (struct) + if len("DealProposal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DealProposal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DealProposal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DealProposal")); err != nil { + return err + } + + if err := t.DealProposal.MarshalCBOR(cw); err != nil { + return err + } + + // t.FundsReceived (big.Int) (struct) + if len("FundsReceived") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"FundsReceived\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("FundsReceived"))); err != nil { + return err + } + if _, err := cw.WriteString(string("FundsReceived")); err != nil { + return err + } + + if err := t.FundsReceived.MarshalCBOR(cw); err != nil { + return err + } + + // t.LegacyProtocol (bool) (bool) + if len("LegacyProtocol") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"LegacyProtocol\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("LegacyProtocol"))); err != nil { + return err + } + if _, err := cw.WriteString(string("LegacyProtocol")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.LegacyProtocol); err != nil { + return err + } + + // t.CurrentInterval (uint64) (uint64) + if len("CurrentInterval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"CurrentInterval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("CurrentInterval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("CurrentInterval")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.CurrentInterval)); err != nil { + return err + } + + return nil +} + +func (t *ProviderDealState) UnmarshalCBOR(r io.Reader) (err error) { + *t = ProviderDealState{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("ProviderDealState: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Status (legacyretrievaltypes.DealStatus) (uint64) + case "Status": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Status = DealStatus(extra) + + } + // t.Message (string) (string) + case "Message": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.StoreID (uint64) (uint64) + case "StoreID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.StoreID = uint64(extra) + + } + // t.Receiver (peer.ID) (string) + case "Receiver": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Receiver = peer.ID(sval) + } + // t.ChannelID (datatransfer.ChannelID) (struct) + case "ChannelID": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.ChannelID = new(datatransfer.ChannelID) + if err := t.ChannelID.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ChannelID pointer: %w", err) + } + } + + } + // t.PieceInfo (piecestore.PieceInfo) (struct) + case "PieceInfo": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.PieceInfo = new(piecestore.PieceInfo) + if err := t.PieceInfo.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PieceInfo pointer: %w", err) + } + } + + } + // t.TotalSent (uint64) (uint64) + case "TotalSent": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.TotalSent = uint64(extra) + + } + // t.DealProposal (legacyretrievaltypes.DealProposal) (struct) + case "DealProposal": + + { + + if err := t.DealProposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DealProposal: %w", err) + } + + } + // t.FundsReceived (big.Int) (struct) + case "FundsReceived": + + { + + if err := t.FundsReceived.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.FundsReceived: %w", err) + } + + } + // t.LegacyProtocol (bool) (bool) + case "LegacyProtocol": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.LegacyProtocol = false + case 21: + t.LegacyProtocol = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.CurrentInterval (uint64) (uint64) + case "CurrentInterval": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.CurrentInterval = uint64(extra) + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *PaymentInfo) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.Lane (uint64) (uint64) + if len("Lane") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Lane\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Lane"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Lane")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Lane)); err != nil { + return err + } + + // t.PayCh (address.Address) (struct) + if len("PayCh") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PayCh\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PayCh"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PayCh")); err != nil { + return err + } + + if err := t.PayCh.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *PaymentInfo) UnmarshalCBOR(r io.Reader) (err error) { + *t = PaymentInfo{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("PaymentInfo: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Lane (uint64) (uint64) + case "Lane": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.Lane = uint64(extra) + + } + // t.PayCh (address.Address) (struct) + case "PayCh": + + { + + if err := t.PayCh.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PayCh: %w", err) + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *RetrievalPeer) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{163}); err != nil { + return err + } + + // t.ID (peer.ID) (string) + if len("ID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ID")); err != nil { + return err + } + + if len(t.ID) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.ID was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.ID))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.ID)); err != nil { + return err + } + + // t.Address (address.Address) (struct) + if len("Address") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Address\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Address"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Address")); err != nil { + return err + } + + if err := t.Address.MarshalCBOR(cw); err != nil { + return err + } + + // t.PieceCID (cid.Cid) (struct) + if len("PieceCID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PieceCID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PieceCID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PieceCID")); err != nil { + return err + } + + if t.PieceCID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PieceCID); err != nil { + return xerrors.Errorf("failed to write cid field t.PieceCID: %w", err) + } + } + + return nil +} + +func (t *RetrievalPeer) UnmarshalCBOR(r io.Reader) (err error) { + *t = RetrievalPeer{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("RetrievalPeer: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.ID (peer.ID) (string) + case "ID": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.ID = peer.ID(sval) + } + // t.Address (address.Address) (struct) + case "Address": + + { + + if err := t.Address.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Address: %w", err) + } + + } + // t.PieceCID (cid.Cid) (struct) + case "PieceCID": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PieceCID: %w", err) + } + + t.PieceCID = &c + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *Ask) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{164}); err != nil { + return err + } + + // t.UnsealPrice (big.Int) (struct) + if len("UnsealPrice") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"UnsealPrice\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("UnsealPrice"))); err != nil { + return err + } + if _, err := cw.WriteString(string("UnsealPrice")); err != nil { + return err + } + + if err := t.UnsealPrice.MarshalCBOR(cw); err != nil { + return err + } + + // t.PricePerByte (big.Int) (struct) + if len("PricePerByte") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PricePerByte\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PricePerByte"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PricePerByte")); err != nil { + return err + } + + if err := t.PricePerByte.MarshalCBOR(cw); err != nil { + return err + } + + // t.PaymentInterval (uint64) (uint64) + if len("PaymentInterval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PaymentInterval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PaymentInterval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PaymentInterval")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PaymentInterval)); err != nil { + return err + } + + // t.PaymentIntervalIncrease (uint64) (uint64) + if len("PaymentIntervalIncrease") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PaymentIntervalIncrease\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PaymentIntervalIncrease"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PaymentIntervalIncrease")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PaymentIntervalIncrease)); err != nil { + return err + } + + return nil +} + +func (t *Ask) UnmarshalCBOR(r io.Reader) (err error) { + *t = Ask{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("Ask: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.UnsealPrice (big.Int) (struct) + case "UnsealPrice": + + { + + if err := t.UnsealPrice.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.UnsealPrice: %w", err) + } + + } + // t.PricePerByte (big.Int) (struct) + case "PricePerByte": + + { + + if err := t.PricePerByte.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.PricePerByte: %w", err) + } + + } + // t.PaymentInterval (uint64) (uint64) + case "PaymentInterval": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PaymentInterval = uint64(extra) + + } + // t.PaymentIntervalIncrease (uint64) (uint64) + case "PaymentIntervalIncrease": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PaymentIntervalIncrease = uint64(extra) + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} diff --git a/retrievalmarket/types/types.go b/retrievalmarket/types/types.go new file mode 100644 index 000000000..929615d81 --- /dev/null +++ b/retrievalmarket/types/types.go @@ -0,0 +1,14 @@ +package types + +import ( + "context" + "io" + + "github.com/filecoin-project/go-state-types/abi" +) + +// SectorAccessor provides methods to unseal and get the seal status of a sector +type SectorAccessor interface { + UnsealSector(ctx context.Context, sectorID abi.SectorNumber, offset abi.UnpaddedPieceSize, length abi.UnpaddedPieceSize) (io.ReadCloser, error) + IsUnsealed(ctx context.Context, sectorID abi.SectorNumber, offset abi.UnpaddedPieceSize, length abi.UnpaddedPieceSize) (bool, error) +} diff --git a/retrievalmarket/types/voucher_legs.go b/retrievalmarket/types/voucher_legs.go index f92747097..b8f629e07 100644 --- a/retrievalmarket/types/voucher_legs.go +++ b/retrievalmarket/types/voucher_legs.go @@ -1,24 +1,18 @@ package types -import ( - datatransfer "github.com/filecoin-project/go-data-transfer" - datatransfer2 "github.com/filecoin-project/go-data-transfer/v2" - "github.com/ipni/go-libipni/dagsync/dtsync" -) - -type LegsVoucherDTv1 struct { - dtsync.Voucher -} - -func (l *LegsVoucherDTv1) Type() datatransfer.TypeIdentifier { - return datatransfer.TypeIdentifier(dtsync.LegsVoucherType) -} - -type LegsVoucherResultDtv1 struct { - VoucherType datatransfer2.TypeIdentifier - dtsync.VoucherResult -} - -func (d *LegsVoucherResultDtv1) Type() datatransfer.TypeIdentifier { - return datatransfer.TypeIdentifier(d.VoucherType) -} +//type LegsVoucherDTv1 struct { +// dtsync.Voucher +//} +// +//func (l *LegsVoucherDTv1) Type() datatransfer.TypeIdentifier { +// return datatransfer.TypeIdentifier(dtsync.LegsVoucherType) +//} +// +//type LegsVoucherResultDtv1 struct { +// VoucherType datatransfer2.TypeIdentifier +// dtsync.VoucherResult +//} +// +//func (d *LegsVoucherResultDtv1) Type() datatransfer.TypeIdentifier { +// return datatransfer.TypeIdentifier(d.VoucherType) +//} diff --git a/storagemarket/dealfilter/cli.go b/storagemarket/dealfilter/cli.go index 96198d1d2..eb4536c75 100644 --- a/storagemarket/dealfilter/cli.go +++ b/storagemarket/dealfilter/cli.go @@ -6,7 +6,7 @@ import ( "encoding/json" "os/exec" - "github.com/filecoin-project/boost-gfm/retrievalmarket" + "github.com/filecoin-project/boost/retrievalmarket/types/legacyretrievaltypes" "github.com/filecoin-project/boost/storagemarket/funds" "github.com/filecoin-project/boost/storagemarket/sealingpipeline" "github.com/filecoin-project/boost/storagemarket/storagespace" @@ -17,7 +17,7 @@ const agent = "boost" const jsonVersion = "2.2.0" type StorageDealFilter func(ctx context.Context, deal DealFilterParams) (bool, string, error) -type RetrievalDealFilter func(ctx context.Context, deal retrievalmarket.ProviderDealState) (bool, string, error) +type RetrievalDealFilter func(ctx context.Context, deal legacyretrievaltypes.ProviderDealState) (bool, string, error) func CliStorageDealFilter(cmd string) StorageDealFilter { return func(ctx context.Context, deal DealFilterParams) (bool, string, error) { @@ -43,9 +43,9 @@ func CliStorageDealFilter(cmd string) StorageDealFilter { } func CliRetrievalDealFilter(cmd string) RetrievalDealFilter { - return func(ctx context.Context, deal retrievalmarket.ProviderDealState) (bool, string, error) { + return func(ctx context.Context, deal legacyretrievaltypes.ProviderDealState) (bool, string, error) { d := struct { - retrievalmarket.ProviderDealState + legacyretrievaltypes.ProviderDealState DealType string FormatVersion string Agent string diff --git a/storagemarket/helper.go b/storagemarket/helper.go index bd693e3ca..8fd1b1446 100644 --- a/storagemarket/helper.go +++ b/storagemarket/helper.go @@ -6,7 +6,7 @@ import ( "errors" "fmt" - "github.com/filecoin-project/boost-gfm/storagemarket" + "github.com/filecoin-project/boost/storagemarket/types" "github.com/filecoin-project/go-state-types/abi" market8 "github.com/filecoin-project/go-state-types/builtin/v9/market" "github.com/filecoin-project/go-state-types/exitcode" @@ -36,7 +36,7 @@ type CurrentDealInfo struct { PublishMsgTipSet ctypes.TipSetKey } -func (c *ChainDealManager) WaitForPublishDeals(ctx context.Context, publishCid cid.Cid, proposal market8.DealProposal) (*storagemarket.PublishDealsWaitResult, error) { +func (c *ChainDealManager) WaitForPublishDeals(ctx context.Context, publishCid cid.Cid, proposal market8.DealProposal) (*types.PublishDealsWaitResult, error) { // Wait for deal to be published (plus additional time for confidence) receipt, err := c.fullnodeApi.StateWaitMsg(ctx, publishCid, c.cfg.PublishDealsConfidence, api.LookbackNoLimit, true) if err != nil { @@ -58,7 +58,7 @@ func (c *ChainDealManager) WaitForPublishDeals(ctx context.Context, publishCid c return nil, fmt.Errorf("WaitForPublishDeals getting deal info errored: %w", err) } - return &storagemarket.PublishDealsWaitResult{DealID: res.DealID, FinalCid: receipt.Message}, nil + return &types.PublishDealsWaitResult{DealID: res.DealID, FinalCid: receipt.Message}, nil } // GetCurrentDealInfo gets the current deal state and deal ID. diff --git a/storagemarket/lp2pimpl/net.go b/storagemarket/lp2pimpl/net.go index c00532d57..dd5683436 100644 --- a/storagemarket/lp2pimpl/net.go +++ b/storagemarket/lp2pimpl/net.go @@ -6,16 +6,16 @@ import ( "fmt" "time" - "github.com/filecoin-project/boost-gfm/shared" - gfm_storagemarket "github.com/filecoin-project/boost-gfm/storagemarket" - gfm_migration "github.com/filecoin-project/boost-gfm/storagemarket/migrations" - gfm_network "github.com/filecoin-project/boost-gfm/storagemarket/network" "github.com/filecoin-project/boost/api" "github.com/filecoin-project/boost/db" + "github.com/filecoin-project/boost/markets/shared" "github.com/filecoin-project/boost/safe" "github.com/filecoin-project/boost/storagemarket" "github.com/filecoin-project/boost/storagemarket/sealingpipeline" "github.com/filecoin-project/boost/storagemarket/types" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" + mig "github.com/filecoin-project/boost/storagemarket/types/legacytypes/migrations" + gfm_network "github.com/filecoin-project/boost/storagemarket/types/legacytypes/network" "github.com/filecoin-project/go-address" cborutil "github.com/filecoin-project/go-cbor-util" "github.com/filecoin-project/go-state-types/crypto" @@ -166,23 +166,21 @@ func NewDealClient(h host.Host, addr address.Address, walletApi api.Wallet, opti // DealProvider listens for incoming deal proposals over libp2p type DealProvider struct { - ctx context.Context - host host.Host - prov *storagemarket.Provider - fullNode v1api.FullNode - plDB *db.ProposalLogsDB - spApi sealingpipeline.API - enableLegacyDeals bool + ctx context.Context + host host.Host + prov *storagemarket.Provider + fullNode v1api.FullNode + plDB *db.ProposalLogsDB + spApi sealingpipeline.API } -func NewDealProvider(h host.Host, prov *storagemarket.Provider, fullNodeApi v1api.FullNode, plDB *db.ProposalLogsDB, spApi sealingpipeline.API, enableLegacyDeals bool) *DealProvider { +func NewDealProvider(h host.Host, prov *storagemarket.Provider, fullNodeApi v1api.FullNode, plDB *db.ProposalLogsDB, spApi sealingpipeline.API) *DealProvider { p := &DealProvider{ - host: h, - prov: prov, - fullNode: fullNodeApi, - plDB: plDB, - spApi: spApi, - enableLegacyDeals: enableLegacyDeals, + host: h, + prov: prov, + fullNode: fullNodeApi, + plDB: plDB, + spApi: spApi, } return p } @@ -206,17 +204,24 @@ func (p *DealProvider) Start(ctx context.Context) { p.host.SetStreamHandler(DealStatusV12ProtocolID, safe.Handle(p.handleNewDealStatusStream)) // Handle legacy deal stream here and reject all legacy deals - if !p.enableLegacyDeals { - p.host.SetStreamHandler(gfm_storagemarket.DealProtocolID101, safe.Handle(p.handleLegacyDealStream)) - p.host.SetStreamHandler(gfm_storagemarket.DealProtocolID110, safe.Handle(p.handleLegacyDealStream)) - p.host.SetStreamHandler(gfm_storagemarket.DealProtocolID111, safe.Handle(p.handleLegacyDealStream)) - } + p.host.SetStreamHandler(legacytypes.DealProtocolID101, safe.Handle(p.handleLegacyDealStream)) + p.host.SetStreamHandler(legacytypes.DealProtocolID110, safe.Handle(p.handleLegacyDealStream)) + p.host.SetStreamHandler(legacytypes.DealProtocolID111, safe.Handle(p.handleLegacyDealStream)) + + // Handle Query Ask + p.host.SetStreamHandler(legacytypes.AskProtocolID, safe.Handle(p.handleNewAskStream)) + p.host.SetStreamHandler(legacytypes.OldAskProtocolID, safe.Handle(p.handleOldAskStream)) } func (p *DealProvider) Stop() { p.host.RemoveStreamHandler(DealProtocolv121ID) p.host.RemoveStreamHandler(DealProtocolv120ID) p.host.RemoveStreamHandler(DealStatusV12ProtocolID) + p.host.RemoveStreamHandler(legacytypes.DealProtocolID101) + p.host.RemoveStreamHandler(legacytypes.DealProtocolID110) + p.host.RemoveStreamHandler(legacytypes.DealProtocolID111) + p.host.RemoveStreamHandler(legacytypes.AskProtocolID) + p.host.RemoveStreamHandler(legacytypes.OldAskProtocolID) } // Called when the client opens a libp2p stream with a new deal proposal @@ -417,13 +422,13 @@ func (p *DealProvider) handleLegacyDealStream(s network.Stream) { rejMsg := fmt.Sprintf("deal proposals made over the legacy %s protocol are deprecated"+ " - please use the %s deal proposal protocol", s.Protocol(), DealProtocolv121ID) - const rejState = gfm_storagemarket.StorageDealProposalRejected + const rejState = 2 var signedResponse typegen.CBORMarshaler _ = s.SetReadDeadline(time.Now().Add(providerReadDeadline)) switch s.Protocol() { - case gfm_storagemarket.DealProtocolID101: - var prop gfm_migration.Proposal0 + case legacytypes.DealProtocolID101: + var prop mig.Proposal0 err := prop.UnmarshalCBOR(s) _ = s.SetReadDeadline(time.Time{}) // Clear read deadline so conn doesn't get closed if err != nil { @@ -437,17 +442,17 @@ func (p *DealProvider) handleLegacyDealStream(s network.Stream) { return } - resp := gfm_migration.Response0{State: rejState, Message: rejMsg, Proposal: pcid} + resp := mig.Response0{State: rejState, Message: rejMsg, Proposal: pcid} sig, err := p.signLegacyResponse(&resp) if err != nil { reqLog.Errorf("getting signed response: %s", err) return } - signedResponse = &gfm_migration.SignedResponse0{Response: resp, Signature: sig} + signedResponse = &mig.SignedResponse0{Response: resp, Signature: sig} - case gfm_storagemarket.DealProtocolID110: - var prop gfm_migration.Proposal1 + case legacytypes.DealProtocolID110: + var prop mig.Proposal1 err := prop.UnmarshalCBOR(s) _ = s.SetReadDeadline(time.Time{}) // Clear read deadline so conn doesn't get closed if err != nil { @@ -470,7 +475,7 @@ func (p *DealProvider) handleLegacyDealStream(s network.Stream) { signedResponse = &gfm_network.SignedResponse{Response: resp, Signature: sig} - case gfm_storagemarket.DealProtocolID111: + case legacytypes.DealProtocolID111: var prop gfm_network.Proposal err := prop.UnmarshalCBOR(s) _ = s.SetReadDeadline(time.Time{}) // Clear read deadline so conn doesn't get closed @@ -533,3 +538,101 @@ func (p *DealProvider) signLegacyResponse(resp typegen.CBORMarshaler) (*crypto.S return localSignature, err } + +func (p *DealProvider) handleNewAskStream(s network.Stream) { + start := time.Now() + reqLog := log.With("client-peer", s.Conn().RemotePeer()) + reqLog.Debugw("new queryAsk request") + + defer func() { + err := s.Close() + if err != nil { + reqLog.Infow("closing stream", "err", err) + } + reqLog.Debugw("handled queryAsk request", "duration", time.Since(start).String()) + }() + + // Read the deal status request from the stream + _ = s.SetReadDeadline(time.Now().Add(providerReadDeadline)) + var req gfm_network.AskRequest + err := req.UnmarshalCBOR(s) + _ = s.SetReadDeadline(time.Time{}) // Clear read deadline so conn doesn't get closed + if err != nil { + reqLog.Warnw("reading queryAsk request from stream", "err", err) + return + } + + var resp gfm_network.AskResponse + + if req.Miner.String() == p.prov.Address.String() { + resp.Ask = p.prov.GetAsk() + } else { + reqLog.Warnw("storage provider for address %s receive ask for miner with address %s", p.prov.Address, req.Miner) + } + + // Set a deadline on writing to the stream so it doesn't hang + _ = s.SetWriteDeadline(time.Now().Add(providerWriteDeadline)) + defer s.SetWriteDeadline(time.Time{}) // nolint + + if err := cborutil.WriteCborRPC(s, &resp); err != nil { + reqLog.Errorw("failed to write queryAsk response", "err", err) + } +} + +func (p *DealProvider) handleOldAskStream(s network.Stream) { + start := time.Now() + reqLog := log.With("client-peer", s.Conn().RemotePeer()) + reqLog.Debugw("new queryAsk request") + + defer func() { + err := s.Close() + if err != nil { + reqLog.Infow("closing stream", "err", err) + } + reqLog.Debugw("handled queryAsk request", "duration", time.Since(start).String()) + }() + + // Read the deal status request from the stream + _ = s.SetReadDeadline(time.Now().Add(providerReadDeadline)) + var req mig.AskRequest0 + err := req.UnmarshalCBOR(s) + _ = s.SetReadDeadline(time.Time{}) // Clear read deadline so conn doesn't get closed + if err != nil { + reqLog.Warnw("reading queryAsk request from stream", "err", err) + return + } + + var resp mig.AskResponse0 + + if req.Miner.String() == p.prov.Address.String() { + ask := p.prov.GetAsk() + + newAsk := ask.Ask + resp.Ask.Ask = &mig.StorageAsk0{ + Price: newAsk.Price, + VerifiedPrice: newAsk.VerifiedPrice, + MinPieceSize: newAsk.MinPieceSize, + MaxPieceSize: newAsk.MaxPieceSize, + Miner: newAsk.Miner, + Timestamp: newAsk.Timestamp, + Expiry: newAsk.Expiry, + SeqNo: newAsk.SeqNo, + } + oldSig, err := p.signLegacyResponse(&resp) + if err != nil { + reqLog.Errorf("getting signed response: %s", err) + } + + resp.Ask.Signature = oldSig + } else { + reqLog.Warnw("storage provider for address %s receive ask for miner with address %s", p.prov.Address, req.Miner) + } + + // Set a deadline on writing to the stream so it doesn't hang + _ = s.SetWriteDeadline(time.Now().Add(providerWriteDeadline)) + defer s.SetWriteDeadline(time.Time{}) // nolint + + if err := cborutil.WriteCborRPC(s, &resp); err != nil { + reqLog.Errorw("failed to write queryAsk response", "err", err) + } +} diff --git a/storagemarket/provider.go b/storagemarket/provider.go index 61b9b415e..379441907 100644 --- a/storagemarket/provider.go +++ b/storagemarket/provider.go @@ -10,7 +10,6 @@ import ( "sync" "time" - "github.com/filecoin-project/boost-gfm/storagemarket" "github.com/filecoin-project/boost/api" "github.com/filecoin-project/boost/build" "github.com/filecoin-project/boost/db" @@ -25,6 +24,7 @@ import ( "github.com/filecoin-project/boost/storagemarket/types" smtypes "github.com/filecoin-project/boost/storagemarket/types" "github.com/filecoin-project/boost/storagemarket/types/dealcheckpoints" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" "github.com/filecoin-project/boost/transport" "github.com/filecoin-project/dagstore" "github.com/filecoin-project/go-address" @@ -57,6 +57,13 @@ type SealingPipelineCache struct { CacheError error } +// PackingResult returns information about how a deal was put into a sector +type PackingResult struct { + SectorNumber abi.SectorNumber + Offset abi.PaddedPieceSize + Size abi.PaddedPieceSize +} + // DagstoreShardRegistry provides the one method from the Dagstore that we use // in deal execution: registering a shard type DagstoreShardRegistry interface { @@ -223,8 +230,8 @@ func (p *Provider) DealBySignedProposalCid(ctx context.Context, propCid cid.Cid) return deal, nil } -func (p *Provider) GetAsk() *storagemarket.SignedStorageAsk { - return p.askGetter.GetAsk() +func (p *Provider) GetAsk() *legacytypes.SignedStorageAsk { + return p.askGetter.GetAsk(p.Address) } // ImportOfflineDealData is called when the Storage Provider imports data for @@ -631,7 +638,7 @@ func (p *Provider) CancelDealDataTransfer(dealUuid uuid.UUID) error { return err } -func (p *Provider) AddPieceToSector(ctx context.Context, deal smtypes.ProviderDealState, pieceData io.Reader) (*storagemarket.PackingResult, error) { +func (p *Provider) AddPieceToSector(ctx context.Context, deal smtypes.ProviderDealState, pieceData io.Reader) (*PackingResult, error) { // Sanity check - we must have published the deal before handing it off // to the sealing subsystem if deal.PublishCID == nil { @@ -657,7 +664,7 @@ func (p *Provider) AddPieceToSector(ctx context.Context, deal smtypes.ProviderDe } p.dealLogger.Infow(deal.DealUuid, "added new deal to sector", "sector", sectorNum.String()) - return &storagemarket.PackingResult{ + return &PackingResult{ SectorNumber: sectorNum, Offset: offset, Size: pieceSize.Padded(), diff --git a/storagemarket/provider_test.go b/storagemarket/provider_test.go index 40443b612..d48418e6a 100644 --- a/storagemarket/provider_test.go +++ b/storagemarket/provider_test.go @@ -16,8 +16,6 @@ import ( "testing" "time" - "github.com/filecoin-project/boost-gfm/shared_testutil" - "github.com/filecoin-project/boost-gfm/storagemarket" "github.com/filecoin-project/boost/db" bdclientutil "github.com/filecoin-project/boost/extern/boostd-data/clientutil" "github.com/filecoin-project/boost/fundmanager" @@ -29,6 +27,7 @@ import ( "github.com/filecoin-project/boost/storagemarket/smtestutil" "github.com/filecoin-project/boost/storagemarket/types" "github.com/filecoin-project/boost/storagemarket/types/dealcheckpoints" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" "github.com/filecoin-project/boost/testutil" "github.com/filecoin-project/boost/transport" "github.com/filecoin-project/boost/transport/httptransport" @@ -851,12 +850,12 @@ func TestDealAskValidation(t *testing.T) { ctx := context.Background() tcs := map[string]struct { - ask *storagemarket.StorageAsk + ask *legacytypes.StorageAsk dbuilder func(h *ProviderHarness) *testDeal expectedErr string }{ "fails if price below minimum for unverified deal": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(100000000000), }, dbuilder: func(h *ProviderHarness) *testDeal { @@ -866,7 +865,7 @@ func TestDealAskValidation(t *testing.T) { expectedErr: "storage price per epoch less than asking price", }, "fails if price below minimum for verified deal": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), VerifiedPrice: abi.NewTokenAmount(100000000000), }, @@ -877,7 +876,7 @@ func TestDealAskValidation(t *testing.T) { expectedErr: "storage price per epoch less than asking price", }, "fails if piece size below minimum": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), MinPieceSize: abi.PaddedPieceSize(1000000000), }, @@ -888,7 +887,7 @@ func TestDealAskValidation(t *testing.T) { expectedErr: "piece size less than minimum required size", }, "fails if piece size above maximum": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), MaxPieceSize: abi.PaddedPieceSize(1), }, @@ -923,14 +922,14 @@ func TestDealVerification(t *testing.T) { ctx := context.Background() tcs := map[string]struct { - ask *storagemarket.StorageAsk + ask *legacytypes.StorageAsk dbuilder func(t *testing.T, h *ProviderHarness) *testDeal expectedErr string expect func(h *ProviderHarness) opts []harnessOpt }{ "fails if client does not have enough datacap for verified deal": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ VerifiedPrice: abi.NewTokenAmount(0), }, dbuilder: func(_ *testing.T, h *ProviderHarness) *testDeal { @@ -944,7 +943,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "verified deal DataCap 1 too small", }, "fails if can't fetch datacap for verified deal": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ VerifiedPrice: abi.NewTokenAmount(0), }, dbuilder: func(_ *testing.T, h *ProviderHarness) *testDeal { @@ -957,7 +956,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "getting verified datacap", }, "fails if client does NOT have enough balance for deal": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(_ *testing.T, h *ProviderHarness) *testDeal { @@ -967,7 +966,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "funds in escrow 0 not enough", }, "fails if client signature is not valid": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(_ *testing.T, h *ProviderHarness) *testDeal { @@ -979,7 +978,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "invalid signature", }, "fails if client signature verification fails": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(_ *testing.T, h *ProviderHarness) *testDeal { @@ -991,7 +990,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "validating signature", }, "fails if proposed provider collateral below minimum": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(_ *testing.T, h *ProviderHarness) *testDeal { @@ -1000,7 +999,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "proposed provider collateral 0 below minimum", }, "fails if proposed provider collateral above maximum": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(_ *testing.T, h *ProviderHarness) *testDeal { @@ -1010,7 +1009,7 @@ func TestDealVerification(t *testing.T) { }, "fails if provider address does not match": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(t *testing.T, h *ProviderHarness) *testDeal { @@ -1021,7 +1020,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "incorrect provider for deal", }, "proposal piece cid has wrong prefix": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(t *testing.T, h *ProviderHarness) *testDeal { @@ -1030,7 +1029,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "proposal PieceCID had wrong prefix", }, "proposal piece cid undefined": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(t *testing.T, h *ProviderHarness) *testDeal { @@ -1039,7 +1038,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "proposal PieceCID undefined", }, "proposal end 9 before proposal start 10": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(t *testing.T, h *ProviderHarness) *testDeal { @@ -1048,7 +1047,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "proposal end 9 before proposal start 10", }, "deal start epoch has already elapsed": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(t *testing.T, h *ProviderHarness) *testDeal { @@ -1057,7 +1056,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "deal start epoch -1 has already elapsed", }, "deal piece size invalid": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(t *testing.T, h *ProviderHarness) *testDeal { @@ -1066,7 +1065,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "proposal piece size is invalid", }, "deal end epoch too far out": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(t *testing.T, h *ProviderHarness) *testDeal { @@ -1077,7 +1076,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "invalid deal end epoch", }, "deal duration greater than max duration": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(t *testing.T, h *ProviderHarness) *testDeal { @@ -1087,7 +1086,7 @@ func TestDealVerification(t *testing.T) { expectedErr: "deal duration out of bounds", }, "deal duration less than min duration": { - ask: &storagemarket.StorageAsk{ + ask: &legacytypes.StorageAsk{ Price: abi.NewTokenAmount(0), }, dbuilder: func(t *testing.T, h *ProviderHarness) *testDeal { @@ -1374,8 +1373,7 @@ type ProviderHarness struct { Transport transport.Transport - SqlDB *sql.DB - DAGStore *shared_testutil.MockDagStoreWrapper + SqlDB *sql.DB } type ChainHeadFn func(ctx context.Context) (*chaintypes.TipSet, error) @@ -2383,21 +2381,20 @@ func (td *testDeal) assertDealFailedNonRecoverable(t *testing.T, ctx context.Con } type mockAskStore struct { - ask *storagemarket.StorageAsk + ask *legacytypes.StorageAsk } func (m *mockAskStore) SetAsk(price, verifiedPrice abi.TokenAmount, minPieceSize, maxPieceSize abi.PaddedPieceSize) { - m.ask = &storagemarket.StorageAsk{ + m.ask = &legacytypes.StorageAsk{ Price: price, VerifiedPrice: verifiedPrice, MinPieceSize: minPieceSize, MaxPieceSize: maxPieceSize, } - } -func (m *mockAskStore) GetAsk() *storagemarket.SignedStorageAsk { - return &storagemarket.SignedStorageAsk{ +func (m *mockAskStore) GetAsk(miner address.Address) *legacytypes.SignedStorageAsk { + return &legacytypes.SignedStorageAsk{ Ask: m.ask, } } diff --git a/storagemarket/smtestutil/mocks.go b/storagemarket/smtestutil/mocks.go index b618c4953..f48dc437c 100644 --- a/storagemarket/smtestutil/mocks.go +++ b/storagemarket/smtestutil/mocks.go @@ -4,12 +4,12 @@ import ( "bytes" "context" "fmt" - "github.com/filecoin-project/go-address" "io" "strings" "sync" - "github.com/filecoin-project/boost-gfm/storagemarket" + "github.com/filecoin-project/go-address" + pdtypes "github.com/filecoin-project/boost/piecedirectory/types" mock_piecedirectory "github.com/filecoin-project/boost/piecedirectory/types/mocks" mock_sealingpipeline "github.com/filecoin-project/boost/storagemarket/sealingpipeline/mock" @@ -131,8 +131,8 @@ func (mb *MinerStubBuilder) SetupNoOp() *MinerStubBuilder { return mb.publishCid, nil }).AnyTimes() - mb.stub.MockChainDealManager.EXPECT().WaitForPublishDeals(gomock.Any(), gomock.Eq(mb.publishCid), gomock.Eq(mb.dp.ClientDealProposal.Proposal)).DoAndReturn(func(_ context.Context, _ cid.Cid, _ market.DealProposal) (*storagemarket.PublishDealsWaitResult, error) { - return &storagemarket.PublishDealsWaitResult{ + mb.stub.MockChainDealManager.EXPECT().WaitForPublishDeals(gomock.Any(), gomock.Eq(mb.publishCid), gomock.Eq(mb.dp.ClientDealProposal.Proposal)).DoAndReturn(func(_ context.Context, _ cid.Cid, _ market.DealProposal) (*types.PublishDealsWaitResult, error) { + return &types.PublishDealsWaitResult{ DealID: mb.dealId, FinalCid: mb.finalPublishCid, }, nil @@ -241,7 +241,7 @@ func (mb *MinerStubBuilder) SetupPublishConfirm(blocking bool) *MinerStubBuilder } mb.stub.lk.Unlock() - mb.stub.MockChainDealManager.EXPECT().WaitForPublishDeals(gomock.Any(), gomock.Eq(mb.publishCid), gomock.Eq(mb.dp.ClientDealProposal.Proposal)).DoAndReturn(func(ctx context.Context, _ cid.Cid, _ market.DealProposal) (*storagemarket.PublishDealsWaitResult, error) { + mb.stub.MockChainDealManager.EXPECT().WaitForPublishDeals(gomock.Any(), gomock.Eq(mb.publishCid), gomock.Eq(mb.dp.ClientDealProposal.Proposal)).DoAndReturn(func(ctx context.Context, _ cid.Cid, _ market.DealProposal) (*types.PublishDealsWaitResult, error) { mb.stub.lk.Lock() ch := mb.stub.unblockWaitForPublish[mb.dp.DealUUID] mb.stub.lk.Unlock() @@ -257,7 +257,7 @@ func (mb *MinerStubBuilder) SetupPublishConfirm(blocking bool) *MinerStubBuilder return nil, ctx.Err() } - return &storagemarket.PublishDealsWaitResult{ + return &types.PublishDealsWaitResult{ DealID: mb.dealId, FinalCid: mb.finalPublishCid, }, nil @@ -267,7 +267,7 @@ func (mb *MinerStubBuilder) SetupPublishConfirm(blocking bool) *MinerStubBuilder } func (mb *MinerStubBuilder) SetupPublishConfirmFailure(err error) *MinerStubBuilder { - mb.stub.MockChainDealManager.EXPECT().WaitForPublishDeals(gomock.Any(), gomock.Eq(mb.publishCid), gomock.Eq(mb.dp.ClientDealProposal.Proposal)).DoAndReturn(func(_ context.Context, _ cid.Cid, _ market.DealProposal) (*storagemarket.PublishDealsWaitResult, error) { + mb.stub.MockChainDealManager.EXPECT().WaitForPublishDeals(gomock.Any(), gomock.Eq(mb.publishCid), gomock.Eq(mb.dp.ClientDealProposal.Proposal)).DoAndReturn(func(_ context.Context, _ cid.Cid, _ market.DealProposal) (*types.PublishDealsWaitResult, error) { return nil, err }) diff --git a/storagemarket/storedask/create_ask_db.sql b/storagemarket/storedask/create_ask_db.sql new file mode 100644 index 000000000..11b48a358 --- /dev/null +++ b/storagemarket/storedask/create_ask_db.sql @@ -0,0 +1,10 @@ +CREATE TABLE IF NOT EXISTS StorageAsk ( + Price INT, + VerifiedPrice INT, + MinPieceSize INT, + MaxPieceSize INT, + Miner Text, + TS INT, + Expiry INT, + SeqNo INT +); \ No newline at end of file diff --git a/storagemarket/storedask/db.go b/storagemarket/storedask/db.go new file mode 100644 index 000000000..ff05a3145 --- /dev/null +++ b/storagemarket/storedask/db.go @@ -0,0 +1,105 @@ +package storedask + +import ( + "context" + "database/sql" + _ "embed" + "errors" + "fmt" + "path" + + "github.com/filecoin-project/boost/db" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" + "github.com/filecoin-project/go-address" + "github.com/filecoin-project/go-state-types/abi" + lotus_repo "github.com/filecoin-project/lotus/node/repo" +) + +const AskDBName = "ask.db" + +//go:embed create_ask_db.sql +var createAskDBSQL string + +func createAskTable(ctx context.Context, askDB *sql.DB) error { + if _, err := askDB.ExecContext(ctx, createAskDBSQL); err != nil { + return fmt.Errorf("failed to create tables in ask DB: %w", err) + } + return nil +} + +type StorageAskDB struct { + db *sql.DB +} + +func NewStorageAskDB(r lotus_repo.LockedRepo) (*StorageAskDB, error) { + dbPath := path.Join(r.Path(), AskDBName+"?cache=shared") + d, err := db.SqlDB(dbPath) + if err != nil { + return nil, err + } + return &StorageAskDB{db: d}, nil +} + +func (s *StorageAskDB) Update(ctx context.Context, ask legacytypes.StorageAsk) error { + var minerString string + qry := "SELECT Miner FROM StorageAsk WHERE Miner=?;" + row := s.db.QueryRowContext(ctx, qry, ask.Miner.String()) + err := row.Scan(&minerString) + switch { + case errors.Is(err, sql.ErrNoRows): + log.Debugf("inserting a new storage ask in db for miner: %s", minerString) + return s.set(ctx, ask) + case err != nil: + return err + default: + log.Debugf("updating the storage ask in db for miner: %s", minerString) + return s.update(ctx, ask) + } +} + +func (s *StorageAskDB) set(ctx context.Context, ask legacytypes.StorageAsk) error { + qry := "INSERT INTO StorageAsk (Price, VerifiedPrice, MinPieceSize, MaxPieceSize, Miner, TS, Expiry, SeqNo) " + qry += "VALUES (?, ?, ?, ?, ?, ?, ?, ?)" + values := []interface{}{ask.Price.Int64(), ask.VerifiedPrice.Int64(), ask.MinPieceSize, ask.MaxPieceSize, ask.Miner.String(), ask.Timestamp, ask.Expiry, ask.SeqNo} + _, err := s.db.ExecContext(ctx, qry, values...) + return err +} + +func (s *StorageAskDB) update(ctx context.Context, ask legacytypes.StorageAsk) error { + qry := "UPDATE StorageAsk SET Price=?, VerifiedPrice=?, MinPieceSize=?, MaxPieceSize=?, TS=?, Expiry=?, SeqNo=? WHERE Miner=?" + values := []interface{}{ask.Price.Int64(), ask.VerifiedPrice.Int64(), ask.MinPieceSize, ask.MaxPieceSize, ask.Timestamp, ask.Expiry, ask.SeqNo, ask.Miner.String()} + _, err := s.db.ExecContext(ctx, qry, values...) + return err +} + +func (s *StorageAskDB) Get(ctx context.Context, miner address.Address) (legacytypes.StorageAsk, error) { + var price, verifiedPrice, timestamp, expiry int64 + var minPieceSize, maxPieceSize, seqNo uint64 + var minerS string + qry := "SELECT Price, VerifiedPrice, MinPieceSize, MaxPieceSize, Miner, TS, Expiry, SeqNo FROM StorageAsk WHERE Miner=?;" + row := s.db.QueryRowContext(ctx, qry, miner.String()) + err := row.Scan(&price, &verifiedPrice, &minPieceSize, &maxPieceSize, &minerS, ×tamp, &expiry, &seqNo) + if err != nil { + return legacytypes.StorageAsk{}, err + } + + m, err := address.NewFromString(minerS) + if err != nil { + return legacytypes.StorageAsk{}, fmt.Errorf("converting stored ask address") + } + + if m != miner { + return legacytypes.StorageAsk{}, fmt.Errorf("stored miner address does match the supplied address") + } + + return legacytypes.StorageAsk{ + Price: abi.NewTokenAmount(price), + VerifiedPrice: abi.NewTokenAmount(verifiedPrice), + Timestamp: abi.ChainEpoch(timestamp), + Expiry: abi.ChainEpoch(expiry), + Miner: miner, + MinPieceSize: abi.PaddedPieceSize(minPieceSize), + MaxPieceSize: abi.PaddedPieceSize(maxPieceSize), + SeqNo: seqNo, + }, nil +} diff --git a/storagemarket/storedask/storedask.go b/storagemarket/storedask/storedask.go new file mode 100644 index 000000000..e547f830c --- /dev/null +++ b/storagemarket/storedask/storedask.go @@ -0,0 +1,226 @@ +package storedask + +import ( + "context" + "database/sql" + "errors" + "fmt" + "sync" + + "github.com/filecoin-project/boost/markets/shared" + "github.com/filecoin-project/boost/node/config" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" + "github.com/filecoin-project/go-address" + cborutil "github.com/filecoin-project/go-cbor-util" + "github.com/filecoin-project/go-state-types/abi" + "github.com/filecoin-project/go-state-types/crypto" + "github.com/filecoin-project/lotus/api" + "github.com/filecoin-project/lotus/chain/types" + logging "github.com/ipfs/go-log/v2" + "go.uber.org/fx" + "golang.org/x/xerrors" +) + +var log = logging.Logger("storedask") + +// DefaultPrice is the default price for unverified deals (in attoFil / GiB / Epoch) +var DefaultPrice = abi.NewTokenAmount(50000000) + +// DefaultVerifiedPrice is the default price for verified deals (in attoFil / GiB / Epoch) +var DefaultVerifiedPrice = abi.NewTokenAmount(5000000) + +// DefaultDuration is the default number of epochs a storage ask is in effect for +const DefaultDuration abi.ChainEpoch = 1000000 + +// DefaultMinPieceSize is the minimum accepted piece size for data +const DefaultMinPieceSize abi.PaddedPieceSize = 256 + +// DefaultMaxPieceSize is the default maximum accepted size for pieces for deals +// TODO: It would be nice to default this to the miner's sector size +const DefaultMaxPieceSize abi.PaddedPieceSize = 32 << 30 + +type StoredAsk interface { + GetAsk(miner address.Address) *legacytypes.SignedStorageAsk + SetAsk(ctx context.Context, price abi.TokenAmount, verifiedPrice abi.TokenAmount, duration abi.ChainEpoch, miner address.Address, options ...legacytypes.StorageAskOption) error +} + +type storedAsk struct { + askLk sync.RWMutex + asks map[address.Address]*legacytypes.SignedStorageAsk + fullNode api.FullNode + db *StorageAskDB +} + +// NewStoredAsk returns a new instance of StoredAsk +// It will initialize a new SignedStorageAsk on disk if one is not set +// Otherwise it loads the current SignedStorageAsk from disk +func NewStoredAsk(cfg *config.Boost) func(lc fx.Lifecycle, askdb *StorageAskDB, fullNode api.FullNode) (*storedAsk, error) { + return func(lc fx.Lifecycle, askdb *StorageAskDB, fullNode api.FullNode) (*storedAsk, error) { + ctx := context.Background() + + err := createAskTable(ctx, askdb.db) + if err != nil { + return nil, err + } + + s := &storedAsk{ + fullNode: fullNode, + db: askdb, + asks: make(map[address.Address]*legacytypes.SignedStorageAsk), + } + + var minerIDs []address.Address + miner, err := address.NewFromString(cfg.Wallets.Miner) + if err != nil { + return nil, fmt.Errorf("converting miner ID from config: %w", err) + } + minerIDs = append(minerIDs, miner) + + for _, m := range minerIDs { + ask, err := s.getSignedAsk(ctx, m) + if err != nil { + if errors.Is(err, sql.ErrNoRows) { + // If not found set everything to default + serr := s.SetAsk(ctx, DefaultPrice, DefaultVerifiedPrice, DefaultDuration, m) + if serr == nil { + continue + } + return nil, fmt.Errorf("setting default ask for miner id %s: %w", m.String(), serr) + } + return nil, fmt.Errorf("failed to initialise AskStore: %w", err) + } + s.asks[m] = &ask + } + + return s, nil + } +} + +func signBytes(ctx context.Context, signer address.Address, b []byte, f api.FullNode) (*crypto.Signature, error) { + signer, err := f.StateAccountKey(ctx, signer, types.EmptyTSK) + if err != nil { + return nil, err + } + + log.Debugf("signing the ask %s with address %s", string(b), signer.String()) + + localSignature, err := f.WalletSign(ctx, signer, b) + if err != nil { + return nil, err + } + return localSignature, nil +} + +func getMinerWorkerAddress(ctx context.Context, maddr address.Address, tok shared.TipSetToken, f api.FullNode) (address.Address, error) { + tsk, err := types.TipSetKeyFromBytes(tok) + if err != nil { + return address.Undef, err + } + + mi, err := f.StateMinerInfo(ctx, maddr, tsk) + if err != nil { + return address.Address{}, err + } + return mi.Worker, nil +} + +func (s *storedAsk) sign(ctx context.Context, ask *legacytypes.StorageAsk) (*crypto.Signature, error) { + tok, err := s.fullNode.ChainHead(ctx) + if err != nil { + return nil, err + } + + return signMinerData(ctx, ask, ask.Miner, tok.Key().Bytes(), s.fullNode) +} + +// SignMinerData signs the given data structure with a signature for the given address +func signMinerData(ctx context.Context, data interface{}, address address.Address, tok shared.TipSetToken, f api.FullNode) (*crypto.Signature, error) { + msg, err := cborutil.Dump(data) + if err != nil { + return nil, xerrors.Errorf("serializing: %w", err) + } + + worker, err := getMinerWorkerAddress(ctx, address, tok, f) + if err != nil { + return nil, err + } + + sig, err := signBytes(ctx, worker, msg, f) + if err != nil { + return nil, xerrors.Errorf("failed to sign: %w", err) + } + return sig, nil +} + +func (s *storedAsk) GetAsk(miner address.Address) *legacytypes.SignedStorageAsk { + s.askLk.RLock() + defer s.askLk.RUnlock() + + return s.asks[miner] +} + +func (s *storedAsk) SetAsk(ctx context.Context, price abi.TokenAmount, verifiedPrice abi.TokenAmount, duration abi.ChainEpoch, miner address.Address, options ...legacytypes.StorageAskOption) error { + s.askLk.Lock() + defer s.askLk.Unlock() + var seqno uint64 + minPieceSize := DefaultMinPieceSize + maxPieceSize := DefaultMaxPieceSize + + oldAsk, ok := s.asks[miner] + if ok { + seqno = oldAsk.Ask.SeqNo + 1 + minPieceSize = oldAsk.Ask.MinPieceSize + maxPieceSize = oldAsk.Ask.MaxPieceSize + } + + ts, err := s.fullNode.ChainHead(ctx) + if err != nil { + return err + } + ask := &legacytypes.StorageAsk{ + Price: price, + VerifiedPrice: verifiedPrice, + Timestamp: ts.Height(), + Expiry: ts.Height() + duration, + Miner: miner, + SeqNo: seqno, + MinPieceSize: minPieceSize, + MaxPieceSize: maxPieceSize, + } + + for _, option := range options { + option(ask) + } + + sig, err := s.sign(ctx, ask) + if err != nil { + return err + } + + s.asks[miner] = &legacytypes.SignedStorageAsk{ + Ask: ask, + Signature: sig, + } + return s.storeAsk(ctx, *ask) + +} + +func (s *storedAsk) getSignedAsk(ctx context.Context, miner address.Address) (legacytypes.SignedStorageAsk, error) { + ask, err := s.db.Get(ctx, miner) + if err != nil { + return legacytypes.SignedStorageAsk{}, err + } + ss, err := s.sign(ctx, &ask) + if err != nil { + return legacytypes.SignedStorageAsk{}, nil + } + + return legacytypes.SignedStorageAsk{ + Ask: &ask, + Signature: ss, + }, nil +} + +func (s *storedAsk) storeAsk(ctx context.Context, ask legacytypes.StorageAsk) error { + return s.db.Update(ctx, ask) +} diff --git a/storagemarket/types/contract_deal_proposal_types_cbor_gen.go b/storagemarket/types/contract_deal_proposal_types_cbor_gen.go index 974e23cba..4a218c407 100644 --- a/storagemarket/types/contract_deal_proposal_types_cbor_gen.go +++ b/storagemarket/types/contract_deal_proposal_types_cbor_gen.go @@ -40,7 +40,7 @@ func (t *ContractParamsVersion1) MarshalCBOR(w io.Writer) error { if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.LocationRef))); err != nil { return err } - if _, err := io.WriteString(w, string(t.LocationRef)); err != nil { + if _, err := cw.WriteString(string(t.LocationRef)); err != nil { return err } diff --git a/storagemarket/types/legacytypes/dealstatus.go b/storagemarket/types/legacytypes/dealstatus.go new file mode 100644 index 000000000..b4fb388fa --- /dev/null +++ b/storagemarket/types/legacytypes/dealstatus.go @@ -0,0 +1,235 @@ +package legacytypes + +import ( + "github.com/filecoin-project/go-statemachine/fsm" +) + +// StorageDealStatus is the local status of a StorageDeal. +// Note: this status has meaning in the context of this module only - it is not +// recorded on chain +type StorageDealStatus = uint64 + +const ( + // StorageDealUnknown means the current status of a deal is undefined + StorageDealUnknown = StorageDealStatus(iota) + + // StorageDealProposalNotFound is a status returned in responses when the deal itself cannot + // be located + StorageDealProposalNotFound + + // StorageDealProposalRejected is returned by a StorageProvider when it chooses not to accept + // a DealProposal + StorageDealProposalRejected + + // StorageDealProposalAccepted indicates an intent to accept a storage deal proposal + StorageDealProposalAccepted + + // StorageDealStaged means a deal has been published and data is ready to be put into a sector + StorageDealStaged + + // StorageDealSealing means a deal is in a sector that is being sealed + StorageDealSealing + + // StorageDealFinalizing means a deal is in a sealed sector and we're doing final + // housekeeping before marking it active + StorageDealFinalizing + + // StorageDealActive means a deal is in a sealed sector and the miner is proving the data + // for the deal + StorageDealActive + + // StorageDealExpired means a deal has passed its final epoch and is expired + StorageDealExpired + + // StorageDealSlashed means the deal was in a sector that got slashed from failing to prove + StorageDealSlashed + + // StorageDealRejecting means the Provider has rejected the deal, and will send a rejection response + StorageDealRejecting + + // StorageDealFailing means something has gone wrong in a deal. Once data is cleaned up the deal will finalize on + // StorageDealError + StorageDealFailing + + // StorageDealFundsReserved means we've deposited funds as necessary to create a deal, ready to move forward + StorageDealFundsReserved + + // StorageDealCheckForAcceptance means the client is waiting for a provider to seal and publish a deal + StorageDealCheckForAcceptance + + // StorageDealValidating means the provider is validating that deal parameters are good for a proposal + StorageDealValidating + + // StorageDealAcceptWait means the provider is running any custom decision logic to decide whether or not to accept the deal + StorageDealAcceptWait + + // StorageDealStartDataTransfer means data transfer is beginning + StorageDealStartDataTransfer + + // StorageDealTransferring means data is being sent from the client to the provider via the data transfer module + StorageDealTransferring + + // StorageDealWaitingForData indicates either a manual transfer + // or that the provider has not received a data transfer request from the client + StorageDealWaitingForData + + // StorageDealVerifyData means data has been transferred and we are attempting to verify it against the PieceCID + StorageDealVerifyData + + // StorageDealReserveProviderFunds means that provider is making sure it has adequate funds for the deal in the StorageMarketActor + StorageDealReserveProviderFunds + + // StorageDealReserveClientFunds means that client is making sure it has adequate funds for the deal in the StorageMarketActor + StorageDealReserveClientFunds + + // StorageDealProviderFunding means that the provider has deposited funds in the StorageMarketActor and it is waiting + // to see the funds appear in its balance + StorageDealProviderFunding + + // StorageDealClientFunding means that the client has deposited funds in the StorageMarketActor and it is waiting + // to see the funds appear in its balance + StorageDealClientFunding + + // StorageDealPublish means the deal is ready to be published on chain + StorageDealPublish + + // StorageDealPublishing means the deal has been published but we are waiting for it to appear on chain + StorageDealPublishing + + // StorageDealError means the deal has failed due to an error, and no further updates will occur + StorageDealError + + // StorageDealProviderTransferAwaitRestart means the provider has restarted while data + // was being transferred from client to provider, and will wait for the client to + // resume the transfer + StorageDealProviderTransferAwaitRestart + + // StorageDealClientTransferRestart means a storage deal data transfer from client to provider will be restarted + // by the client + StorageDealClientTransferRestart + + // StorageDealAwaitingPreCommit means a deal is ready and must be pre-committed + StorageDealAwaitingPreCommit + + // StorageDealTransferQueued means the data transfer request has been queued and will be executed soon. + StorageDealTransferQueued +) + +// DealStates maps StorageDealStatus codes to string names +var DealStates = map[StorageDealStatus]string{ + StorageDealUnknown: "StorageDealUnknown", + StorageDealProposalNotFound: "StorageDealProposalNotFound", + StorageDealProposalRejected: "StorageDealProposalRejected", + StorageDealProposalAccepted: "StorageDealProposalAccepted", + StorageDealAcceptWait: "StorageDealAcceptWait", + StorageDealStartDataTransfer: "StorageDealStartDataTransfer", + StorageDealStaged: "StorageDealStaged", + StorageDealAwaitingPreCommit: "StorageDealAwaitingPreCommit", + StorageDealSealing: "StorageDealSealing", + StorageDealActive: "StorageDealActive", + StorageDealExpired: "StorageDealExpired", + StorageDealSlashed: "StorageDealSlashed", + StorageDealRejecting: "StorageDealRejecting", + StorageDealFailing: "StorageDealFailing", + StorageDealFundsReserved: "StorageDealFundsReserved", + StorageDealCheckForAcceptance: "StorageDealCheckForAcceptance", + StorageDealValidating: "StorageDealValidating", + StorageDealTransferring: "StorageDealTransferring", + StorageDealWaitingForData: "StorageDealWaitingForData", + StorageDealVerifyData: "StorageDealVerifyData", + StorageDealReserveProviderFunds: "StorageDealReserveProviderFunds", + StorageDealReserveClientFunds: "StorageDealReserveClientFunds", + StorageDealProviderFunding: "StorageDealProviderFunding", + StorageDealClientFunding: "StorageDealClientFunding", + StorageDealPublish: "StorageDealPublish", + StorageDealPublishing: "StorageDealPublishing", + StorageDealError: "StorageDealError", + StorageDealFinalizing: "StorageDealFinalizing", + StorageDealClientTransferRestart: "StorageDealClientTransferRestart", + StorageDealProviderTransferAwaitRestart: "StorageDealProviderTransferAwaitRestart", + StorageDealTransferQueued: "StorageDealTransferQueued", +} + +// DealStatesDescriptions maps StorageDealStatus codes to string description for better UX +var DealStatesDescriptions = map[StorageDealStatus]string{ + StorageDealUnknown: "Unknown", + StorageDealProposalNotFound: "Proposal not found", + StorageDealProposalRejected: "Proposal rejected", + StorageDealProposalAccepted: "Proposal accepted", + StorageDealAcceptWait: "AcceptWait", + StorageDealStartDataTransfer: "Starting data transfer", + StorageDealStaged: "Staged", + StorageDealAwaitingPreCommit: "Awaiting a PreCommit message on chain", + StorageDealSealing: "Sealing", + StorageDealActive: "Active", + StorageDealExpired: "Expired", + StorageDealSlashed: "Slashed", + StorageDealRejecting: "Rejecting", + StorageDealFailing: "Failing", + StorageDealFundsReserved: "FundsReserved", + StorageDealCheckForAcceptance: "Checking for deal acceptance", + StorageDealValidating: "Validating", + StorageDealTransferring: "Transferring", + StorageDealWaitingForData: "Waiting for data", + StorageDealVerifyData: "Verifying data", + StorageDealReserveProviderFunds: "Reserving provider funds", + StorageDealReserveClientFunds: "Reserving client funds", + StorageDealProviderFunding: "Provider funding", + StorageDealClientFunding: "Client funding", + StorageDealPublish: "Publish", + StorageDealPublishing: "Publishing", + StorageDealError: "Error", + StorageDealFinalizing: "Finalizing", + StorageDealClientTransferRestart: "Client transfer restart", + StorageDealProviderTransferAwaitRestart: "ProviderTransferAwaitRestart", +} + +var DealStatesDurations = map[StorageDealStatus]string{ + StorageDealUnknown: "", + StorageDealProposalNotFound: "", + StorageDealProposalRejected: "", + StorageDealProposalAccepted: "a few minutes", + StorageDealAcceptWait: "a few minutes", + StorageDealStartDataTransfer: "a few minutes", + StorageDealStaged: "a few minutes", + StorageDealAwaitingPreCommit: "a few minutes", + StorageDealSealing: "a few hours", + StorageDealActive: "", + StorageDealExpired: "", + StorageDealSlashed: "", + StorageDealRejecting: "", + StorageDealFailing: "", + StorageDealFundsReserved: "a few minutes", + StorageDealCheckForAcceptance: "a few minutes", + StorageDealValidating: "a few minutes", + StorageDealTransferring: "a few minutes", + StorageDealWaitingForData: "a few minutes", + StorageDealVerifyData: "a few minutes", + StorageDealReserveProviderFunds: "a few minutes", + StorageDealReserveClientFunds: "a few minutes", + StorageDealProviderFunding: "a few minutes", + StorageDealClientFunding: "a few minutes", + StorageDealPublish: "a few minutes", + StorageDealPublishing: "a few minutes", + StorageDealError: "", + StorageDealFinalizing: "a few minutes", + StorageDealClientTransferRestart: "depending on data size, anywhere between a few minutes to a few hours", + StorageDealProviderTransferAwaitRestart: "a few minutes", +} + +// ProviderFinalityStates are the states that terminate deal processing for a deal. +// When a provider restarts, it restarts only deals that are not in a finality state. +var ProviderFinalityStates = []fsm.StateKey{ + StorageDealError, + StorageDealSlashed, + StorageDealExpired, +} + +// StatesKnownBySealingSubsystem are the states on the happy path after hand-off to +// the sealing subsystem +var StatesKnownBySealingSubsystem = []fsm.StateKey{ + StorageDealAwaitingPreCommit, + StorageDealSealing, + StorageDealFinalizing, + StorageDealActive, +} diff --git a/storagemarket/types/legacytypes/filestore/file.go b/storagemarket/types/legacytypes/filestore/file.go new file mode 100644 index 000000000..119ced0f1 --- /dev/null +++ b/storagemarket/types/legacytypes/filestore/file.go @@ -0,0 +1,39 @@ +package filestore + +import ( + "os" + "path" +) + +type fd struct { + *os.File + filename string + basepath string +} + +func newFile(basepath OsPath, filename Path) (File, error) { + var err error + result := fd{filename: string(filename), basepath: string(basepath)} + full := path.Join(string(basepath), string(filename)) + result.File, err = os.OpenFile(full, os.O_APPEND|os.O_CREATE|os.O_RDWR, 0644) + if err != nil { + return nil, err + } + return &result, nil +} + +func (f fd) Path() Path { + return Path(f.filename) +} + +func (f fd) OsPath() OsPath { + return OsPath(f.Name()) +} + +func (f fd) Size() int64 { + info, err := os.Stat(f.Name()) + if err != nil { + return -1 + } + return info.Size() +} diff --git a/storagemarket/types/legacytypes/filestore/filestore.go b/storagemarket/types/legacytypes/filestore/filestore.go new file mode 100644 index 000000000..9334bc230 --- /dev/null +++ b/storagemarket/types/legacytypes/filestore/filestore.go @@ -0,0 +1,82 @@ +package filestore + +import ( + "fmt" + "io" + "os" + "path" + "path/filepath" +) + +type fileStore struct { + base string +} + +// NewLocalFileStore creates a filestore mounted on a given local directory path +func NewLocalFileStore(baseDir OsPath) (FileStore, error) { + base, err := checkIsDir(string(baseDir)) + if err != nil { + return nil, err + } + return &fileStore{base}, nil +} + +func (fs fileStore) filename(p Path) string { + return filepath.Join(fs.base, string(p)) +} + +func (fs fileStore) Open(p Path) (File, error) { + name := fs.filename(p) + if _, err := os.Stat(name); err != nil { + return nil, fmt.Errorf("error trying to open %s: %s", name, err.Error()) + } + return newFile(OsPath(fs.base), p) +} + +func (fs fileStore) Create(p Path) (File, error) { + name := fs.filename(p) + if _, err := os.Stat(name); err == nil { + return nil, fmt.Errorf("file %s already exists", name) + } + return newFile(OsPath(fs.base), p) +} + +func (fs fileStore) Store(p Path, src File) (Path, error) { + dest, err := fs.Create(p) + if err != nil { + return Path(""), err + } + + if _, err = io.Copy(dest, src); err != nil { + dest.Close() + return Path(""), err + } + return p, dest.Close() +} + +func (fs fileStore) Delete(p Path) error { + filename := string(p) + full := path.Join(string(fs.base), string(filename)) + return os.Remove(full) +} + +func (fs fileStore) CreateTemp() (File, error) { + f, err := os.CreateTemp(fs.base, "fstmp") + if err != nil { + return nil, err + } + filename := filepath.Base(f.Name()) + return &fd{File: f, basepath: fs.base, filename: filename}, nil +} + +func checkIsDir(baseDir string) (string, error) { + base := filepath.Clean(string(baseDir)) + info, err := os.Stat(base) + if err != nil { + return "", fmt.Errorf("error getting %s info: %s", base, err.Error()) + } + if !info.IsDir() { + return "", fmt.Errorf("%s is not a directory", base) + } + return base, nil +} diff --git a/storagemarket/types/legacytypes/filestore/types.go b/storagemarket/types/legacytypes/filestore/types.go new file mode 100644 index 000000000..d3e840319 --- /dev/null +++ b/storagemarket/types/legacytypes/filestore/types.go @@ -0,0 +1,38 @@ +package filestore + +import ( + "io" +) + +// Path represents an abstract path to a file +type Path string + +// OsPath represents a path that can be located on +// the operating system with standard os.File operations +type OsPath string + +// File is a wrapper around an os file +type File interface { + Path() Path + OsPath() OsPath + Size() int64 + + io.Closer + io.Reader + io.Writer + io.Seeker +} + +// FileStore is an abstract filestore, used for storing temporary file data +// when handing off a deal to the Storage Mining module. Files are created by +// the storage market module, their path is given to the storage mining module +// when AddPiece is called. The Storage Mining module then reads from them +// from the FileStore, and deletes them once they have been sealed in a sector +type FileStore interface { + Open(p Path) (File, error) + Create(p Path) (File, error) + Store(p Path, f File) (Path, error) + Delete(p Path) error + + CreateTemp() (File, error) +} diff --git a/storagemarket/types/legacytypes/migrations/migrations.go b/storagemarket/types/legacytypes/migrations/migrations.go new file mode 100644 index 000000000..5a9ba6752 --- /dev/null +++ b/storagemarket/types/legacytypes/migrations/migrations.go @@ -0,0 +1,325 @@ +package migrations + +import ( + "fmt" + "unicode/utf8" + + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" + "github.com/ipfs/go-cid" + "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + "golang.org/x/net/context" + + "github.com/filecoin-project/go-address" + versioning "github.com/filecoin-project/go-ds-versioning/pkg" + "github.com/filecoin-project/go-ds-versioning/pkg/versioned" + "github.com/filecoin-project/go-state-types/abi" + "github.com/filecoin-project/go-state-types/builtin/v9/market" + "github.com/filecoin-project/go-state-types/crypto" + marketOld "github.com/filecoin-project/specs-actors/actors/builtin/market" + + "github.com/filecoin-project/boost/storagemarket/types/legacytypes/filestore" +) + +//go:generate cbor-gen-for ClientDeal0 MinerDeal0 Balance0 SignedStorageAsk0 StorageAsk0 DataRef0 ProviderDealState0 AskRequest0 AskResponse0 Proposal0 Response0 SignedResponse0 DealStatusRequest0 DealStatusResponse0 + +// Balance0 is version 0 of Balance +type Balance0 struct { + Locked abi.TokenAmount + Available abi.TokenAmount +} + +// StorageAsk0 is version 0 of StorageAsk +type StorageAsk0 struct { + Price abi.TokenAmount + VerifiedPrice abi.TokenAmount + + MinPieceSize abi.PaddedPieceSize + MaxPieceSize abi.PaddedPieceSize + Miner address.Address + Timestamp abi.ChainEpoch + Expiry abi.ChainEpoch + SeqNo uint64 +} + +// SignedStorageAsk0 is version 0 of SignedStorageAsk +type SignedStorageAsk0 struct { + Ask *StorageAsk0 + Signature *crypto.Signature +} + +// MinerDeal0 is version 0 of MinerDeal +type MinerDeal0 struct { + marketOld.ClientDealProposal + ProposalCid cid.Cid + AddFundsCid *cid.Cid + PublishCid *cid.Cid + Miner peer.ID + Client peer.ID + State legacytypes.StorageDealStatus + PiecePath filestore.Path + MetadataPath filestore.Path + SlashEpoch abi.ChainEpoch + FastRetrieval bool + Message string + StoreID *uint64 + FundsReserved abi.TokenAmount + Ref *DataRef0 + AvailableForRetrieval bool + + DealID abi.DealID + CreationTime cbg.CborTime +} + +// ClientDeal0 is version 0 of ClientDeal +type ClientDeal0 struct { + market.ClientDealProposal + ProposalCid cid.Cid + AddFundsCid *cid.Cid + State legacytypes.StorageDealStatus + Miner peer.ID + MinerWorker address.Address + DealID abi.DealID + DataRef *DataRef0 + Message string + PublishMessage *cid.Cid + SlashEpoch abi.ChainEpoch + PollRetryCount uint64 + PollErrorCount uint64 + FastRetrieval bool + StoreID *uint64 + FundsReserved abi.TokenAmount + CreationTime cbg.CborTime +} + +// DataRef0 is version 0 of DataRef +type DataRef0 struct { + TransferType string + Root cid.Cid + PieceCid *cid.Cid + PieceSize abi.UnpaddedPieceSize +} + +// ProviderDealState0 is version 0 of ProviderDealState +type ProviderDealState0 struct { + State legacytypes.StorageDealStatus + Message string + Proposal *market.DealProposal + ProposalCid *cid.Cid + AddFundsCid *cid.Cid + PublishCid *cid.Cid + DealID abi.DealID + FastRetrieval bool +} + +// Proposal0 is version 0 of Proposal +type Proposal0 struct { + DealProposal *market.ClientDealProposal + Piece *DataRef0 + FastRetrieval bool +} + +// Response0 is version 0 of Response +type Response0 struct { + State legacytypes.StorageDealStatus + + // DealProposalRejected + Message string + Proposal cid.Cid + + // StorageDealProposalAccepted + PublishMessage *cid.Cid +} + +// SignedResponse0 is version 0 of SignedResponse +type SignedResponse0 struct { + Response Response0 + Signature *crypto.Signature +} + +// AskRequest0 is version 0 of AskRequest +type AskRequest0 struct { + Miner address.Address +} + +// AskResponse0 is version 0 of AskResponse +type AskResponse0 struct { + Ask *SignedStorageAsk0 +} + +// DealStatusRequest0 is version 0 of DealStatusRequest +type DealStatusRequest0 struct { + Proposal cid.Cid + Signature crypto.Signature +} + +// DealStatusResponse0 is version 0 of DealStatusResponse +type DealStatusResponse0 struct { + DealState ProviderDealState0 + Signature crypto.Signature +} + +// MigrateDataRef0To1 migrates a tuple encoded data tref to a map encoded data ref +func MigrateDataRef0To1(oldDr *DataRef0) *legacytypes.DataRef { + if oldDr == nil { + return nil + } + return &legacytypes.DataRef{ + TransferType: oldDr.TransferType, + Root: oldDr.Root, + PieceCid: oldDr.PieceCid, + PieceSize: oldDr.PieceSize, + } +} + +// MigrateClientDeal0To1 migrates a tuple encoded client deal to a map encoded client deal +func MigrateClientDeal0To1(oldCd *ClientDeal0) (*legacytypes.ClientDeal, error) { + return &legacytypes.ClientDeal{ + ClientDealProposal: oldCd.ClientDealProposal, + ProposalCid: oldCd.ProposalCid, + AddFundsCid: oldCd.AddFundsCid, + State: oldCd.State, + Miner: oldCd.Miner, + MinerWorker: oldCd.MinerWorker, + DealID: oldCd.DealID, + DataRef: MigrateDataRef0To1(oldCd.DataRef), + Message: oldCd.Message, + PublishMessage: oldCd.PublishMessage, + SlashEpoch: oldCd.SlashEpoch, + PollRetryCount: oldCd.PollRetryCount, + PollErrorCount: oldCd.PollErrorCount, + FastRetrieval: oldCd.FastRetrieval, + FundsReserved: oldCd.FundsReserved, + CreationTime: oldCd.CreationTime, + }, nil +} + +// MigrateMinerDeal0To1 migrates a tuple encoded miner deal to a map encoded miner deal +func MigrateMinerDeal0To1(oldCd *MinerDeal0) (*MinerDeal1, error) { + return &MinerDeal1{ + ClientDealProposal: oldCd.ClientDealProposal, + ProposalCid: oldCd.ProposalCid, + AddFundsCid: oldCd.AddFundsCid, + PublishCid: oldCd.PublishCid, + Miner: oldCd.Miner, + Client: oldCd.Client, + State: oldCd.State, + PiecePath: oldCd.PiecePath, + MetadataPath: oldCd.MetadataPath, + SlashEpoch: oldCd.SlashEpoch, + FastRetrieval: oldCd.FastRetrieval, + Message: oldCd.Message, + FundsReserved: oldCd.FundsReserved, + Ref: MigrateDataRef0To1(oldCd.Ref), + AvailableForRetrieval: oldCd.AvailableForRetrieval, + DealID: oldCd.DealID, + CreationTime: oldCd.CreationTime, + }, nil +} + +// MigrateMinerDeal1To2 migrates a miner deal label to the new format +func MigrateMinerDeal1To2(oldCd *MinerDeal1) (*legacytypes.MinerDeal, error) { + clientDealProp, err := MigrateClientDealProposal0To1(oldCd.ClientDealProposal) + if err != nil { + return nil, fmt.Errorf("migrating deal with proposal cid %s: %w", oldCd.ProposalCid, err) + } + + return &legacytypes.MinerDeal{ + ClientDealProposal: *clientDealProp, + ProposalCid: oldCd.ProposalCid, + AddFundsCid: oldCd.AddFundsCid, + PublishCid: oldCd.PublishCid, + Miner: oldCd.Miner, + Client: oldCd.Client, + State: oldCd.State, + PiecePath: oldCd.PiecePath, + MetadataPath: oldCd.MetadataPath, + SlashEpoch: oldCd.SlashEpoch, + FastRetrieval: oldCd.FastRetrieval, + Message: oldCd.Message, + FundsReserved: oldCd.FundsReserved, + Ref: oldCd.Ref, + AvailableForRetrieval: oldCd.AvailableForRetrieval, + DealID: oldCd.DealID, + CreationTime: oldCd.CreationTime, + }, nil +} + +func MigrateClientDealProposal0To1(prop marketOld.ClientDealProposal) (*legacytypes.ClientDealProposal, error) { + oldLabel := prop.Proposal.Label + + var err error + var newLabel market.DealLabel + if utf8.ValidString(oldLabel) { + newLabel, err = market.NewLabelFromString(oldLabel) + if err != nil { + return nil, fmt.Errorf("migrating deal label to DealLabel (string): %w", err) + } + } else { + newLabel, err = market.NewLabelFromBytes([]byte(oldLabel)) + if err != nil { + return nil, fmt.Errorf("migrating deal label to DealLabel (byte): %w", err) + } + } + + return &legacytypes.ClientDealProposal{ + ClientSignature: prop.ClientSignature, + Proposal: market.DealProposal{ + PieceCID: prop.Proposal.PieceCID, + PieceSize: prop.Proposal.PieceSize, + VerifiedDeal: prop.Proposal.VerifiedDeal, + Client: prop.Proposal.Client, + Provider: prop.Proposal.Provider, + Label: newLabel, + StartEpoch: prop.Proposal.StartEpoch, + EndEpoch: prop.Proposal.EndEpoch, + StoragePricePerEpoch: prop.Proposal.StoragePricePerEpoch, + ProviderCollateral: prop.Proposal.ProviderCollateral, + ClientCollateral: prop.Proposal.ClientCollateral, + }, + }, nil +} + +// MigrateStorageAsk0To1 migrates a tuple encoded storage ask to a map encoded storage ask +func MigrateStorageAsk0To1(oldSa *StorageAsk0) *legacytypes.StorageAsk { + return &legacytypes.StorageAsk{ + Price: oldSa.Price, + VerifiedPrice: oldSa.VerifiedPrice, + + MinPieceSize: oldSa.MinPieceSize, + MaxPieceSize: oldSa.MaxPieceSize, + Miner: oldSa.Miner, + Timestamp: oldSa.Timestamp, + Expiry: oldSa.Expiry, + SeqNo: oldSa.SeqNo, + } +} + +// GetMigrateSignedStorageAsk0To1 returns a function that migrates a tuple encoded signed storage ask to a map encoded signed storage ask +// It needs a signing function to resign the ask -- there's no way around that +func GetMigrateSignedStorageAsk0To1(sign func(ctx context.Context, ask *legacytypes.StorageAsk) (*crypto.Signature, error)) func(*SignedStorageAsk0) (*legacytypes.SignedStorageAsk, error) { + return func(oldSsa *SignedStorageAsk0) (*legacytypes.SignedStorageAsk, error) { + newSa := MigrateStorageAsk0To1(oldSsa.Ask) + sig, err := sign(context.TODO(), newSa) + if err != nil { + return nil, err + } + return &legacytypes.SignedStorageAsk{ + Ask: newSa, + Signature: sig, + }, nil + } +} + +// ClientMigrations are migrations for the client's store of storage deals +var ClientMigrations = versioned.BuilderList{ + versioned.NewVersionedBuilder(MigrateClientDeal0To1, versioning.VersionKey("1")), +} + +// ProviderMigrations are migrations for the providers's store of storage deals +var ProviderMigrations = versioned.BuilderList{ + versioned.NewVersionedBuilder(MigrateMinerDeal0To1, versioning.VersionKey("1")).FilterKeys([]string{ + "/latest-ask", "/storage-ask/latest", "/storage-ask/1/latest", "/storage-ask/versions/current"}), + versioned.NewVersionedBuilder(MigrateMinerDeal1To2, versioning.VersionKey("2")).FilterKeys([]string{ + "/latest-ask", "/storage-ask/latest", "/storage-ask/1/latest", "/storage-ask/versions/current"}).OldVersion("1"), +} diff --git a/storagemarket/types/legacytypes/migrations/migrations_cbor_gen.go b/storagemarket/types/legacytypes/migrations/migrations_cbor_gen.go new file mode 100644 index 000000000..6610cfec8 --- /dev/null +++ b/storagemarket/types/legacytypes/migrations/migrations_cbor_gen.go @@ -0,0 +1,2271 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package migrations + +import ( + "fmt" + "io" + "math" + "sort" + + filestore "github.com/filecoin-project/boost/storagemarket/types/legacytypes/filestore" + abi "github.com/filecoin-project/go-state-types/abi" + market "github.com/filecoin-project/go-state-types/builtin/v9/market" + crypto "github.com/filecoin-project/go-state-types/crypto" + cid "github.com/ipfs/go-cid" + peer "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + xerrors "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +var lengthBufClientDeal0 = []byte{145} + +func (t *ClientDeal0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufClientDeal0); err != nil { + return err + } + + // t.ClientDealProposal (market.ClientDealProposal) (struct) + if err := t.ClientDealProposal.MarshalCBOR(cw); err != nil { + return err + } + + // t.ProposalCid (cid.Cid) (struct) + + if err := cbg.WriteCid(cw, t.ProposalCid); err != nil { + return xerrors.Errorf("failed to write cid field t.ProposalCid: %w", err) + } + + // t.AddFundsCid (cid.Cid) (struct) + + if t.AddFundsCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.AddFundsCid); err != nil { + return xerrors.Errorf("failed to write cid field t.AddFundsCid: %w", err) + } + } + + // t.State (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.State)); err != nil { + return err + } + + // t.Miner (peer.ID) (string) + if len(t.Miner) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Miner was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Miner))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Miner)); err != nil { + return err + } + + // t.MinerWorker (address.Address) (struct) + if err := t.MinerWorker.MarshalCBOR(cw); err != nil { + return err + } + + // t.DealID (abi.DealID) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.DealID)); err != nil { + return err + } + + // t.DataRef (migrations.DataRef0) (struct) + if err := t.DataRef.MarshalCBOR(cw); err != nil { + return err + } + + // t.Message (string) (string) + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.PublishMessage (cid.Cid) (struct) + + if t.PublishMessage == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PublishMessage); err != nil { + return xerrors.Errorf("failed to write cid field t.PublishMessage: %w", err) + } + } + + // t.SlashEpoch (abi.ChainEpoch) (int64) + if t.SlashEpoch >= 0 { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.SlashEpoch)); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajNegativeInt, uint64(-t.SlashEpoch-1)); err != nil { + return err + } + } + + // t.PollRetryCount (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PollRetryCount)); err != nil { + return err + } + + // t.PollErrorCount (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PollErrorCount)); err != nil { + return err + } + + // t.FastRetrieval (bool) (bool) + if err := cbg.WriteBool(w, t.FastRetrieval); err != nil { + return err + } + + // t.StoreID (uint64) (uint64) + + if t.StoreID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(*t.StoreID)); err != nil { + return err + } + } + + // t.FundsReserved (big.Int) (struct) + if err := t.FundsReserved.MarshalCBOR(cw); err != nil { + return err + } + + // t.CreationTime (typegen.CborTime) (struct) + if err := t.CreationTime.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *ClientDeal0) UnmarshalCBOR(r io.Reader) (err error) { + *t = ClientDeal0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 17 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.ClientDealProposal (market.ClientDealProposal) (struct) + + { + + if err := t.ClientDealProposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ClientDealProposal: %w", err) + } + + } + // t.ProposalCid (cid.Cid) (struct) + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.ProposalCid: %w", err) + } + + t.ProposalCid = c + + } + // t.AddFundsCid (cid.Cid) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.AddFundsCid: %w", err) + } + + t.AddFundsCid = &c + } + + } + // t.State (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.State = uint64(extra) + + } + // t.Miner (peer.ID) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Miner = peer.ID(sval) + } + // t.MinerWorker (address.Address) (struct) + + { + + if err := t.MinerWorker.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.MinerWorker: %w", err) + } + + } + // t.DealID (abi.DealID) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.DealID = abi.DealID(extra) + + } + // t.DataRef (migrations.DataRef0) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.DataRef = new(DataRef0) + if err := t.DataRef.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DataRef pointer: %w", err) + } + } + + } + // t.Message (string) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.PublishMessage (cid.Cid) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PublishMessage: %w", err) + } + + t.PublishMessage = &c + } + + } + // t.SlashEpoch (abi.ChainEpoch) (int64) + { + maj, extra, err := cr.ReadHeader() + var extraI int64 + if err != nil { + return err + } + switch maj { + case cbg.MajUnsignedInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 positive overflow") + } + case cbg.MajNegativeInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 negative overflow") + } + extraI = -1 - extraI + default: + return fmt.Errorf("wrong type for int64 field: %d", maj) + } + + t.SlashEpoch = abi.ChainEpoch(extraI) + } + // t.PollRetryCount (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PollRetryCount = uint64(extra) + + } + // t.PollErrorCount (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PollErrorCount = uint64(extra) + + } + // t.FastRetrieval (bool) (bool) + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.FastRetrieval = false + case 21: + t.FastRetrieval = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.StoreID (uint64) (uint64) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + typed := uint64(extra) + t.StoreID = &typed + } + + } + // t.FundsReserved (big.Int) (struct) + + { + + if err := t.FundsReserved.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.FundsReserved: %w", err) + } + + } + // t.CreationTime (typegen.CborTime) (struct) + + { + + if err := t.CreationTime.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.CreationTime: %w", err) + } + + } + return nil +} + +var lengthBufMinerDeal0 = []byte{146} + +func (t *MinerDeal0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufMinerDeal0); err != nil { + return err + } + + // t.ClientDealProposal (market.ClientDealProposal) (struct) + if err := t.ClientDealProposal.MarshalCBOR(cw); err != nil { + return err + } + + // t.ProposalCid (cid.Cid) (struct) + + if err := cbg.WriteCid(cw, t.ProposalCid); err != nil { + return xerrors.Errorf("failed to write cid field t.ProposalCid: %w", err) + } + + // t.AddFundsCid (cid.Cid) (struct) + + if t.AddFundsCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.AddFundsCid); err != nil { + return xerrors.Errorf("failed to write cid field t.AddFundsCid: %w", err) + } + } + + // t.PublishCid (cid.Cid) (struct) + + if t.PublishCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PublishCid); err != nil { + return xerrors.Errorf("failed to write cid field t.PublishCid: %w", err) + } + } + + // t.Miner (peer.ID) (string) + if len(t.Miner) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Miner was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Miner))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Miner)); err != nil { + return err + } + + // t.Client (peer.ID) (string) + if len(t.Client) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Client was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Client))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Client)); err != nil { + return err + } + + // t.State (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.State)); err != nil { + return err + } + + // t.PiecePath (filestore.Path) (string) + if len(t.PiecePath) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.PiecePath was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.PiecePath))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.PiecePath)); err != nil { + return err + } + + // t.MetadataPath (filestore.Path) (string) + if len(t.MetadataPath) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.MetadataPath was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.MetadataPath))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.MetadataPath)); err != nil { + return err + } + + // t.SlashEpoch (abi.ChainEpoch) (int64) + if t.SlashEpoch >= 0 { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.SlashEpoch)); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajNegativeInt, uint64(-t.SlashEpoch-1)); err != nil { + return err + } + } + + // t.FastRetrieval (bool) (bool) + if err := cbg.WriteBool(w, t.FastRetrieval); err != nil { + return err + } + + // t.Message (string) (string) + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.StoreID (uint64) (uint64) + + if t.StoreID == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(*t.StoreID)); err != nil { + return err + } + } + + // t.FundsReserved (big.Int) (struct) + if err := t.FundsReserved.MarshalCBOR(cw); err != nil { + return err + } + + // t.Ref (migrations.DataRef0) (struct) + if err := t.Ref.MarshalCBOR(cw); err != nil { + return err + } + + // t.AvailableForRetrieval (bool) (bool) + if err := cbg.WriteBool(w, t.AvailableForRetrieval); err != nil { + return err + } + + // t.DealID (abi.DealID) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.DealID)); err != nil { + return err + } + + // t.CreationTime (typegen.CborTime) (struct) + if err := t.CreationTime.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *MinerDeal0) UnmarshalCBOR(r io.Reader) (err error) { + *t = MinerDeal0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 18 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.ClientDealProposal (market.ClientDealProposal) (struct) + + { + + if err := t.ClientDealProposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ClientDealProposal: %w", err) + } + + } + // t.ProposalCid (cid.Cid) (struct) + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.ProposalCid: %w", err) + } + + t.ProposalCid = c + + } + // t.AddFundsCid (cid.Cid) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.AddFundsCid: %w", err) + } + + t.AddFundsCid = &c + } + + } + // t.PublishCid (cid.Cid) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PublishCid: %w", err) + } + + t.PublishCid = &c + } + + } + // t.Miner (peer.ID) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Miner = peer.ID(sval) + } + // t.Client (peer.ID) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Client = peer.ID(sval) + } + // t.State (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.State = uint64(extra) + + } + // t.PiecePath (filestore.Path) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.PiecePath = filestore.Path(sval) + } + // t.MetadataPath (filestore.Path) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.MetadataPath = filestore.Path(sval) + } + // t.SlashEpoch (abi.ChainEpoch) (int64) + { + maj, extra, err := cr.ReadHeader() + var extraI int64 + if err != nil { + return err + } + switch maj { + case cbg.MajUnsignedInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 positive overflow") + } + case cbg.MajNegativeInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 negative overflow") + } + extraI = -1 - extraI + default: + return fmt.Errorf("wrong type for int64 field: %d", maj) + } + + t.SlashEpoch = abi.ChainEpoch(extraI) + } + // t.FastRetrieval (bool) (bool) + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.FastRetrieval = false + case 21: + t.FastRetrieval = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.Message (string) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.StoreID (uint64) (uint64) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + typed := uint64(extra) + t.StoreID = &typed + } + + } + // t.FundsReserved (big.Int) (struct) + + { + + if err := t.FundsReserved.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.FundsReserved: %w", err) + } + + } + // t.Ref (migrations.DataRef0) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Ref = new(DataRef0) + if err := t.Ref.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Ref pointer: %w", err) + } + } + + } + // t.AvailableForRetrieval (bool) (bool) + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.AvailableForRetrieval = false + case 21: + t.AvailableForRetrieval = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.DealID (abi.DealID) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.DealID = abi.DealID(extra) + + } + // t.CreationTime (typegen.CborTime) (struct) + + { + + if err := t.CreationTime.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.CreationTime: %w", err) + } + + } + return nil +} + +var lengthBufBalance0 = []byte{130} + +func (t *Balance0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufBalance0); err != nil { + return err + } + + // t.Locked (big.Int) (struct) + if err := t.Locked.MarshalCBOR(cw); err != nil { + return err + } + + // t.Available (big.Int) (struct) + if err := t.Available.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *Balance0) UnmarshalCBOR(r io.Reader) (err error) { + *t = Balance0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 2 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Locked (big.Int) (struct) + + { + + if err := t.Locked.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Locked: %w", err) + } + + } + // t.Available (big.Int) (struct) + + { + + if err := t.Available.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Available: %w", err) + } + + } + return nil +} + +var lengthBufSignedStorageAsk0 = []byte{130} + +func (t *SignedStorageAsk0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufSignedStorageAsk0); err != nil { + return err + } + + // t.Ask (migrations.StorageAsk0) (struct) + if err := t.Ask.MarshalCBOR(cw); err != nil { + return err + } + + // t.Signature (crypto.Signature) (struct) + if err := t.Signature.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *SignedStorageAsk0) UnmarshalCBOR(r io.Reader) (err error) { + *t = SignedStorageAsk0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 2 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Ask (migrations.StorageAsk0) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Ask = new(StorageAsk0) + if err := t.Ask.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Ask pointer: %w", err) + } + } + + } + // t.Signature (crypto.Signature) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Signature = new(crypto.Signature) + if err := t.Signature.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Signature pointer: %w", err) + } + } + + } + return nil +} + +var lengthBufStorageAsk0 = []byte{136} + +func (t *StorageAsk0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufStorageAsk0); err != nil { + return err + } + + // t.Price (big.Int) (struct) + if err := t.Price.MarshalCBOR(cw); err != nil { + return err + } + + // t.VerifiedPrice (big.Int) (struct) + if err := t.VerifiedPrice.MarshalCBOR(cw); err != nil { + return err + } + + // t.MinPieceSize (abi.PaddedPieceSize) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.MinPieceSize)); err != nil { + return err + } + + // t.MaxPieceSize (abi.PaddedPieceSize) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.MaxPieceSize)); err != nil { + return err + } + + // t.Miner (address.Address) (struct) + if err := t.Miner.MarshalCBOR(cw); err != nil { + return err + } + + // t.Timestamp (abi.ChainEpoch) (int64) + if t.Timestamp >= 0 { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Timestamp)); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajNegativeInt, uint64(-t.Timestamp-1)); err != nil { + return err + } + } + + // t.Expiry (abi.ChainEpoch) (int64) + if t.Expiry >= 0 { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Expiry)); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajNegativeInt, uint64(-t.Expiry-1)); err != nil { + return err + } + } + + // t.SeqNo (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.SeqNo)); err != nil { + return err + } + + return nil +} + +func (t *StorageAsk0) UnmarshalCBOR(r io.Reader) (err error) { + *t = StorageAsk0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 8 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Price (big.Int) (struct) + + { + + if err := t.Price.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Price: %w", err) + } + + } + // t.VerifiedPrice (big.Int) (struct) + + { + + if err := t.VerifiedPrice.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.VerifiedPrice: %w", err) + } + + } + // t.MinPieceSize (abi.PaddedPieceSize) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.MinPieceSize = abi.PaddedPieceSize(extra) + + } + // t.MaxPieceSize (abi.PaddedPieceSize) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.MaxPieceSize = abi.PaddedPieceSize(extra) + + } + // t.Miner (address.Address) (struct) + + { + + if err := t.Miner.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Miner: %w", err) + } + + } + // t.Timestamp (abi.ChainEpoch) (int64) + { + maj, extra, err := cr.ReadHeader() + var extraI int64 + if err != nil { + return err + } + switch maj { + case cbg.MajUnsignedInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 positive overflow") + } + case cbg.MajNegativeInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 negative overflow") + } + extraI = -1 - extraI + default: + return fmt.Errorf("wrong type for int64 field: %d", maj) + } + + t.Timestamp = abi.ChainEpoch(extraI) + } + // t.Expiry (abi.ChainEpoch) (int64) + { + maj, extra, err := cr.ReadHeader() + var extraI int64 + if err != nil { + return err + } + switch maj { + case cbg.MajUnsignedInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 positive overflow") + } + case cbg.MajNegativeInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 negative overflow") + } + extraI = -1 - extraI + default: + return fmt.Errorf("wrong type for int64 field: %d", maj) + } + + t.Expiry = abi.ChainEpoch(extraI) + } + // t.SeqNo (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.SeqNo = uint64(extra) + + } + return nil +} + +var lengthBufDataRef0 = []byte{132} + +func (t *DataRef0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufDataRef0); err != nil { + return err + } + + // t.TransferType (string) (string) + if len(t.TransferType) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.TransferType was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.TransferType))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.TransferType)); err != nil { + return err + } + + // t.Root (cid.Cid) (struct) + + if err := cbg.WriteCid(cw, t.Root); err != nil { + return xerrors.Errorf("failed to write cid field t.Root: %w", err) + } + + // t.PieceCid (cid.Cid) (struct) + + if t.PieceCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PieceCid); err != nil { + return xerrors.Errorf("failed to write cid field t.PieceCid: %w", err) + } + } + + // t.PieceSize (abi.UnpaddedPieceSize) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PieceSize)); err != nil { + return err + } + + return nil +} + +func (t *DataRef0) UnmarshalCBOR(r io.Reader) (err error) { + *t = DataRef0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 4 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.TransferType (string) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.TransferType = string(sval) + } + // t.Root (cid.Cid) (struct) + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.Root: %w", err) + } + + t.Root = c + + } + // t.PieceCid (cid.Cid) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PieceCid: %w", err) + } + + t.PieceCid = &c + } + + } + // t.PieceSize (abi.UnpaddedPieceSize) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PieceSize = abi.UnpaddedPieceSize(extra) + + } + return nil +} + +var lengthBufProviderDealState0 = []byte{136} + +func (t *ProviderDealState0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufProviderDealState0); err != nil { + return err + } + + // t.State (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.State)); err != nil { + return err + } + + // t.Message (string) (string) + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.Proposal (market.DealProposal) (struct) + if err := t.Proposal.MarshalCBOR(cw); err != nil { + return err + } + + // t.ProposalCid (cid.Cid) (struct) + + if t.ProposalCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.ProposalCid); err != nil { + return xerrors.Errorf("failed to write cid field t.ProposalCid: %w", err) + } + } + + // t.AddFundsCid (cid.Cid) (struct) + + if t.AddFundsCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.AddFundsCid); err != nil { + return xerrors.Errorf("failed to write cid field t.AddFundsCid: %w", err) + } + } + + // t.PublishCid (cid.Cid) (struct) + + if t.PublishCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PublishCid); err != nil { + return xerrors.Errorf("failed to write cid field t.PublishCid: %w", err) + } + } + + // t.DealID (abi.DealID) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.DealID)); err != nil { + return err + } + + // t.FastRetrieval (bool) (bool) + if err := cbg.WriteBool(w, t.FastRetrieval); err != nil { + return err + } + return nil +} + +func (t *ProviderDealState0) UnmarshalCBOR(r io.Reader) (err error) { + *t = ProviderDealState0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 8 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.State (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.State = uint64(extra) + + } + // t.Message (string) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.Proposal (market.DealProposal) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Proposal = new(market.DealProposal) + if err := t.Proposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Proposal pointer: %w", err) + } + } + + } + // t.ProposalCid (cid.Cid) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.ProposalCid: %w", err) + } + + t.ProposalCid = &c + } + + } + // t.AddFundsCid (cid.Cid) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.AddFundsCid: %w", err) + } + + t.AddFundsCid = &c + } + + } + // t.PublishCid (cid.Cid) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PublishCid: %w", err) + } + + t.PublishCid = &c + } + + } + // t.DealID (abi.DealID) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.DealID = abi.DealID(extra) + + } + // t.FastRetrieval (bool) (bool) + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.FastRetrieval = false + case 21: + t.FastRetrieval = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + return nil +} + +var lengthBufAskRequest0 = []byte{129} + +func (t *AskRequest0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufAskRequest0); err != nil { + return err + } + + // t.Miner (address.Address) (struct) + if err := t.Miner.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *AskRequest0) UnmarshalCBOR(r io.Reader) (err error) { + *t = AskRequest0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 1 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Miner (address.Address) (struct) + + { + + if err := t.Miner.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Miner: %w", err) + } + + } + return nil +} + +var lengthBufAskResponse0 = []byte{129} + +func (t *AskResponse0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufAskResponse0); err != nil { + return err + } + + // t.Ask (migrations.SignedStorageAsk0) (struct) + if err := t.Ask.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *AskResponse0) UnmarshalCBOR(r io.Reader) (err error) { + *t = AskResponse0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 1 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Ask (migrations.SignedStorageAsk0) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Ask = new(SignedStorageAsk0) + if err := t.Ask.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Ask pointer: %w", err) + } + } + + } + return nil +} + +var lengthBufProposal0 = []byte{131} + +func (t *Proposal0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufProposal0); err != nil { + return err + } + + // t.DealProposal (market.ClientDealProposal) (struct) + if err := t.DealProposal.MarshalCBOR(cw); err != nil { + return err + } + + // t.Piece (migrations.DataRef0) (struct) + if err := t.Piece.MarshalCBOR(cw); err != nil { + return err + } + + // t.FastRetrieval (bool) (bool) + if err := cbg.WriteBool(w, t.FastRetrieval); err != nil { + return err + } + return nil +} + +func (t *Proposal0) UnmarshalCBOR(r io.Reader) (err error) { + *t = Proposal0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 3 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.DealProposal (market.ClientDealProposal) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.DealProposal = new(market.ClientDealProposal) + if err := t.DealProposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DealProposal pointer: %w", err) + } + } + + } + // t.Piece (migrations.DataRef0) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Piece = new(DataRef0) + if err := t.Piece.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Piece pointer: %w", err) + } + } + + } + // t.FastRetrieval (bool) (bool) + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.FastRetrieval = false + case 21: + t.FastRetrieval = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + return nil +} + +var lengthBufResponse0 = []byte{132} + +func (t *Response0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufResponse0); err != nil { + return err + } + + // t.State (uint64) (uint64) + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.State)); err != nil { + return err + } + + // t.Message (string) (string) + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.Proposal (cid.Cid) (struct) + + if err := cbg.WriteCid(cw, t.Proposal); err != nil { + return xerrors.Errorf("failed to write cid field t.Proposal: %w", err) + } + + // t.PublishMessage (cid.Cid) (struct) + + if t.PublishMessage == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PublishMessage); err != nil { + return xerrors.Errorf("failed to write cid field t.PublishMessage: %w", err) + } + } + + return nil +} + +func (t *Response0) UnmarshalCBOR(r io.Reader) (err error) { + *t = Response0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 4 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.State (uint64) (uint64) + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.State = uint64(extra) + + } + // t.Message (string) (string) + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.Proposal (cid.Cid) (struct) + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.Proposal: %w", err) + } + + t.Proposal = c + + } + // t.PublishMessage (cid.Cid) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PublishMessage: %w", err) + } + + t.PublishMessage = &c + } + + } + return nil +} + +var lengthBufSignedResponse0 = []byte{130} + +func (t *SignedResponse0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufSignedResponse0); err != nil { + return err + } + + // t.Response (migrations.Response0) (struct) + if err := t.Response.MarshalCBOR(cw); err != nil { + return err + } + + // t.Signature (crypto.Signature) (struct) + if err := t.Signature.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *SignedResponse0) UnmarshalCBOR(r io.Reader) (err error) { + *t = SignedResponse0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 2 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Response (migrations.Response0) (struct) + + { + + if err := t.Response.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Response: %w", err) + } + + } + // t.Signature (crypto.Signature) (struct) + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Signature = new(crypto.Signature) + if err := t.Signature.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Signature pointer: %w", err) + } + } + + } + return nil +} + +var lengthBufDealStatusRequest0 = []byte{130} + +func (t *DealStatusRequest0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufDealStatusRequest0); err != nil { + return err + } + + // t.Proposal (cid.Cid) (struct) + + if err := cbg.WriteCid(cw, t.Proposal); err != nil { + return xerrors.Errorf("failed to write cid field t.Proposal: %w", err) + } + + // t.Signature (crypto.Signature) (struct) + if err := t.Signature.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *DealStatusRequest0) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealStatusRequest0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 2 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.Proposal (cid.Cid) (struct) + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.Proposal: %w", err) + } + + t.Proposal = c + + } + // t.Signature (crypto.Signature) (struct) + + { + + if err := t.Signature.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Signature: %w", err) + } + + } + return nil +} + +var lengthBufDealStatusResponse0 = []byte{130} + +func (t *DealStatusResponse0) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write(lengthBufDealStatusResponse0); err != nil { + return err + } + + // t.DealState (migrations.ProviderDealState0) (struct) + if err := t.DealState.MarshalCBOR(cw); err != nil { + return err + } + + // t.Signature (crypto.Signature) (struct) + if err := t.Signature.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *DealStatusResponse0) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealStatusResponse0{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajArray { + return fmt.Errorf("cbor input should be of type array") + } + + if extra != 2 { + return fmt.Errorf("cbor input had wrong number of fields") + } + + // t.DealState (migrations.ProviderDealState0) (struct) + + { + + if err := t.DealState.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DealState: %w", err) + } + + } + // t.Signature (crypto.Signature) (struct) + + { + + if err := t.Signature.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Signature: %w", err) + } + + } + return nil +} diff --git a/storagemarket/types/legacytypes/migrations/migrations_mapenc_types.go b/storagemarket/types/legacytypes/migrations/migrations_mapenc_types.go new file mode 100644 index 000000000..84bc456be --- /dev/null +++ b/storagemarket/types/legacytypes/migrations/migrations_mapenc_types.go @@ -0,0 +1,55 @@ +package migrations + +import ( + "github.com/filecoin-project/boost/datatransfer" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" + "github.com/ipfs/go-cid" + "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + + "github.com/filecoin-project/go-state-types/abi" + marketOld "github.com/filecoin-project/specs-actors/actors/builtin/market" + + "github.com/filecoin-project/boost/storagemarket/types/legacytypes/filestore" +) + +// Some of the types in the migrations file are CBOR array-encoded, and some +// are map-encoded. The --map-encoding parameter must be specified in a +// generate directive in a separate file. So we define CBOR map-encoded types +// in this file + +//go:generate cbor-gen-for --map-encoding Proposal1 MinerDeal1 + +// Proposal1 is version 1 of Proposal (used by deal proposal protocol v1.1.0) +type Proposal1 struct { + DealProposal *marketOld.ClientDealProposal + Piece *legacytypes.DataRef + FastRetrieval bool +} + +// MinerDeal1 is version 1 of MinerDeal +type MinerDeal1 struct { + marketOld.ClientDealProposal + ProposalCid cid.Cid + AddFundsCid *cid.Cid + PublishCid *cid.Cid + Miner peer.ID + Client peer.ID + State legacytypes.StorageDealStatus + PiecePath filestore.Path + MetadataPath filestore.Path + SlashEpoch abi.ChainEpoch + FastRetrieval bool + Message string + FundsReserved abi.TokenAmount + Ref *legacytypes.DataRef + AvailableForRetrieval bool + + DealID abi.DealID + CreationTime cbg.CborTime + + TransferChannelId *datatransfer.ChannelID + SectorNumber abi.SectorNumber + + InboundCAR string +} diff --git a/storagemarket/types/legacytypes/migrations/migrations_mapenc_types_cbor_gen.go b/storagemarket/types/legacytypes/migrations/migrations_mapenc_types_cbor_gen.go new file mode 100644 index 000000000..56dbc9fc3 --- /dev/null +++ b/storagemarket/types/legacytypes/migrations/migrations_mapenc_types_cbor_gen.go @@ -0,0 +1,936 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package migrations + +import ( + "fmt" + "io" + "math" + "sort" + + datatransfer "github.com/filecoin-project/boost/datatransfer" + legacytypes "github.com/filecoin-project/boost/storagemarket/types/legacytypes" + filestore "github.com/filecoin-project/boost/storagemarket/types/legacytypes/filestore" + abi "github.com/filecoin-project/go-state-types/abi" + market "github.com/filecoin-project/specs-actors/actors/builtin/market" + cid "github.com/ipfs/go-cid" + peer "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + xerrors "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +func (t *Proposal1) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{163}); err != nil { + return err + } + + // t.Piece (legacytypes.DataRef) (struct) + if len("Piece") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Piece\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Piece"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Piece")); err != nil { + return err + } + + if err := t.Piece.MarshalCBOR(cw); err != nil { + return err + } + + // t.DealProposal (market.ClientDealProposal) (struct) + if len("DealProposal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DealProposal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DealProposal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DealProposal")); err != nil { + return err + } + + if err := t.DealProposal.MarshalCBOR(cw); err != nil { + return err + } + + // t.FastRetrieval (bool) (bool) + if len("FastRetrieval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"FastRetrieval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("FastRetrieval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("FastRetrieval")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.FastRetrieval); err != nil { + return err + } + return nil +} + +func (t *Proposal1) UnmarshalCBOR(r io.Reader) (err error) { + *t = Proposal1{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("Proposal1: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Piece (legacytypes.DataRef) (struct) + case "Piece": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Piece = new(legacytypes.DataRef) + if err := t.Piece.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Piece pointer: %w", err) + } + } + + } + // t.DealProposal (market.ClientDealProposal) (struct) + case "DealProposal": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.DealProposal = new(market.ClientDealProposal) + if err := t.DealProposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DealProposal pointer: %w", err) + } + } + + } + // t.FastRetrieval (bool) (bool) + case "FastRetrieval": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.FastRetrieval = false + case 21: + t.FastRetrieval = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *MinerDeal1) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{180}); err != nil { + return err + } + + // t.Ref (legacytypes.DataRef) (struct) + if len("Ref") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Ref\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Ref"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Ref")); err != nil { + return err + } + + if err := t.Ref.MarshalCBOR(cw); err != nil { + return err + } + + // t.Miner (peer.ID) (string) + if len("Miner") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Miner\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Miner"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Miner")); err != nil { + return err + } + + if len(t.Miner) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Miner was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Miner))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Miner)); err != nil { + return err + } + + // t.State (uint64) (uint64) + if len("State") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"State\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("State"))); err != nil { + return err + } + if _, err := cw.WriteString(string("State")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.State)); err != nil { + return err + } + + // t.Client (peer.ID) (string) + if len("Client") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Client\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Client"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Client")); err != nil { + return err + } + + if len(t.Client) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Client was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Client))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Client)); err != nil { + return err + } + + // t.DealID (abi.DealID) (uint64) + if len("DealID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DealID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DealID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DealID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.DealID)); err != nil { + return err + } + + // t.Message (string) (string) + if len("Message") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Message\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Message"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Message")); err != nil { + return err + } + + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.PiecePath (filestore.Path) (string) + if len("PiecePath") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PiecePath\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PiecePath"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PiecePath")); err != nil { + return err + } + + if len(t.PiecePath) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.PiecePath was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.PiecePath))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.PiecePath)); err != nil { + return err + } + + // t.InboundCAR (string) (string) + if len("InboundCAR") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"InboundCAR\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("InboundCAR"))); err != nil { + return err + } + if _, err := cw.WriteString(string("InboundCAR")); err != nil { + return err + } + + if len(t.InboundCAR) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.InboundCAR was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.InboundCAR))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.InboundCAR)); err != nil { + return err + } + + // t.PublishCid (cid.Cid) (struct) + if len("PublishCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PublishCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PublishCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PublishCid")); err != nil { + return err + } + + if t.PublishCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PublishCid); err != nil { + return xerrors.Errorf("failed to write cid field t.PublishCid: %w", err) + } + } + + // t.SlashEpoch (abi.ChainEpoch) (int64) + if len("SlashEpoch") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"SlashEpoch\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("SlashEpoch"))); err != nil { + return err + } + if _, err := cw.WriteString(string("SlashEpoch")); err != nil { + return err + } + + if t.SlashEpoch >= 0 { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.SlashEpoch)); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajNegativeInt, uint64(-t.SlashEpoch-1)); err != nil { + return err + } + } + + // t.AddFundsCid (cid.Cid) (struct) + if len("AddFundsCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"AddFundsCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("AddFundsCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("AddFundsCid")); err != nil { + return err + } + + if t.AddFundsCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.AddFundsCid); err != nil { + return xerrors.Errorf("failed to write cid field t.AddFundsCid: %w", err) + } + } + + // t.ProposalCid (cid.Cid) (struct) + if len("ProposalCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ProposalCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ProposalCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ProposalCid")); err != nil { + return err + } + + if err := cbg.WriteCid(cw, t.ProposalCid); err != nil { + return xerrors.Errorf("failed to write cid field t.ProposalCid: %w", err) + } + + // t.CreationTime (typegen.CborTime) (struct) + if len("CreationTime") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"CreationTime\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("CreationTime"))); err != nil { + return err + } + if _, err := cw.WriteString(string("CreationTime")); err != nil { + return err + } + + if err := t.CreationTime.MarshalCBOR(cw); err != nil { + return err + } + + // t.MetadataPath (filestore.Path) (string) + if len("MetadataPath") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"MetadataPath\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("MetadataPath"))); err != nil { + return err + } + if _, err := cw.WriteString(string("MetadataPath")); err != nil { + return err + } + + if len(t.MetadataPath) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.MetadataPath was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.MetadataPath))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.MetadataPath)); err != nil { + return err + } + + // t.SectorNumber (abi.SectorNumber) (uint64) + if len("SectorNumber") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"SectorNumber\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("SectorNumber"))); err != nil { + return err + } + if _, err := cw.WriteString(string("SectorNumber")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.SectorNumber)); err != nil { + return err + } + + // t.FastRetrieval (bool) (bool) + if len("FastRetrieval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"FastRetrieval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("FastRetrieval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("FastRetrieval")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.FastRetrieval); err != nil { + return err + } + + // t.FundsReserved (big.Int) (struct) + if len("FundsReserved") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"FundsReserved\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("FundsReserved"))); err != nil { + return err + } + if _, err := cw.WriteString(string("FundsReserved")); err != nil { + return err + } + + if err := t.FundsReserved.MarshalCBOR(cw); err != nil { + return err + } + + // t.TransferChannelId (datatransfer.ChannelID) (struct) + if len("TransferChannelId") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"TransferChannelId\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("TransferChannelId"))); err != nil { + return err + } + if _, err := cw.WriteString(string("TransferChannelId")); err != nil { + return err + } + + if err := t.TransferChannelId.MarshalCBOR(cw); err != nil { + return err + } + + // t.ClientDealProposal (market.ClientDealProposal) (struct) + if len("ClientDealProposal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ClientDealProposal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ClientDealProposal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ClientDealProposal")); err != nil { + return err + } + + if err := t.ClientDealProposal.MarshalCBOR(cw); err != nil { + return err + } + + // t.AvailableForRetrieval (bool) (bool) + if len("AvailableForRetrieval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"AvailableForRetrieval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("AvailableForRetrieval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("AvailableForRetrieval")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.AvailableForRetrieval); err != nil { + return err + } + return nil +} + +func (t *MinerDeal1) UnmarshalCBOR(r io.Reader) (err error) { + *t = MinerDeal1{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("MinerDeal1: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Ref (legacytypes.DataRef) (struct) + case "Ref": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Ref = new(legacytypes.DataRef) + if err := t.Ref.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Ref pointer: %w", err) + } + } + + } + // t.Miner (peer.ID) (string) + case "Miner": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Miner = peer.ID(sval) + } + // t.State (uint64) (uint64) + case "State": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.State = uint64(extra) + + } + // t.Client (peer.ID) (string) + case "Client": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Client = peer.ID(sval) + } + // t.DealID (abi.DealID) (uint64) + case "DealID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.DealID = abi.DealID(extra) + + } + // t.Message (string) (string) + case "Message": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.PiecePath (filestore.Path) (string) + case "PiecePath": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.PiecePath = filestore.Path(sval) + } + // t.InboundCAR (string) (string) + case "InboundCAR": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.InboundCAR = string(sval) + } + // t.PublishCid (cid.Cid) (struct) + case "PublishCid": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PublishCid: %w", err) + } + + t.PublishCid = &c + } + + } + // t.SlashEpoch (abi.ChainEpoch) (int64) + case "SlashEpoch": + { + maj, extra, err := cr.ReadHeader() + var extraI int64 + if err != nil { + return err + } + switch maj { + case cbg.MajUnsignedInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 positive overflow") + } + case cbg.MajNegativeInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 negative overflow") + } + extraI = -1 - extraI + default: + return fmt.Errorf("wrong type for int64 field: %d", maj) + } + + t.SlashEpoch = abi.ChainEpoch(extraI) + } + // t.AddFundsCid (cid.Cid) (struct) + case "AddFundsCid": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.AddFundsCid: %w", err) + } + + t.AddFundsCid = &c + } + + } + // t.ProposalCid (cid.Cid) (struct) + case "ProposalCid": + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.ProposalCid: %w", err) + } + + t.ProposalCid = c + + } + // t.CreationTime (typegen.CborTime) (struct) + case "CreationTime": + + { + + if err := t.CreationTime.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.CreationTime: %w", err) + } + + } + // t.MetadataPath (filestore.Path) (string) + case "MetadataPath": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.MetadataPath = filestore.Path(sval) + } + // t.SectorNumber (abi.SectorNumber) (uint64) + case "SectorNumber": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.SectorNumber = abi.SectorNumber(extra) + + } + // t.FastRetrieval (bool) (bool) + case "FastRetrieval": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.FastRetrieval = false + case 21: + t.FastRetrieval = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.FundsReserved (big.Int) (struct) + case "FundsReserved": + + { + + if err := t.FundsReserved.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.FundsReserved: %w", err) + } + + } + // t.TransferChannelId (datatransfer.ChannelID) (struct) + case "TransferChannelId": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.TransferChannelId = new(datatransfer.ChannelID) + if err := t.TransferChannelId.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.TransferChannelId pointer: %w", err) + } + } + + } + // t.ClientDealProposal (market.ClientDealProposal) (struct) + case "ClientDealProposal": + + { + + if err := t.ClientDealProposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ClientDealProposal: %w", err) + } + + } + // t.AvailableForRetrieval (bool) (bool) + case "AvailableForRetrieval": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.AvailableForRetrieval = false + case 21: + t.AvailableForRetrieval = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} diff --git a/storagemarket/types/legacytypes/network/types.go b/storagemarket/types/legacytypes/network/types.go new file mode 100644 index 000000000..604e0801f --- /dev/null +++ b/storagemarket/types/legacytypes/network/types.go @@ -0,0 +1,79 @@ +package network + +import ( + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" + "github.com/filecoin-project/go-address" + "github.com/filecoin-project/go-state-types/builtin/v9/market" + "github.com/filecoin-project/go-state-types/crypto" + "github.com/ipfs/go-cid" +) + +//go:generate cbor-gen-for --map-encoding AskRequest AskResponse Proposal Response SignedResponse DealStatusRequest DealStatusResponse + +// Proposal is the data sent over the network from client to provider when proposing +// a deal +type Proposal struct { + DealProposal *market.ClientDealProposal + Piece *legacytypes.DataRef + FastRetrieval bool +} + +// ProposalUndefined is an empty Proposal message +var ProposalUndefined = Proposal{} + +// Response is a response to a proposal sent over the network +type Response struct { + State legacytypes.StorageDealStatus + + // DealProposalRejected + Message string + Proposal cid.Cid + + // StorageDealProposalAccepted + PublishMessage *cid.Cid +} + +// SignedResponse is a response that is signed +type SignedResponse struct { + Response Response + + Signature *crypto.Signature +} + +// SignedResponseUndefined represents an empty SignedResponse message +var SignedResponseUndefined = SignedResponse{} + +// AskRequest is a request for current ask parameters for a given miner +type AskRequest struct { + Miner address.Address +} + +// AskRequestUndefined represents and empty AskRequest message +var AskRequestUndefined = AskRequest{} + +// AskResponse is the response sent over the network in response +// to an ask request +type AskResponse struct { + Ask *legacytypes.SignedStorageAsk +} + +// AskResponseUndefined represents an empty AskResponse message +var AskResponseUndefined = AskResponse{} + +// DealStatusRequest sent by a client to query deal status +type DealStatusRequest struct { + Proposal cid.Cid + Signature crypto.Signature +} + +// DealStatusRequestUndefined represents an empty DealStatusRequest message +var DealStatusRequestUndefined = DealStatusRequest{} + +// DealStatusResponse is a provider's response to DealStatusRequest +type DealStatusResponse struct { + DealState legacytypes.ProviderDealState + Signature crypto.Signature +} + +// DealStatusResponseUndefined represents an empty DealStatusResponse message +var DealStatusResponseUndefined = DealStatusResponse{} diff --git a/storagemarket/types/legacytypes/network/types_cbor_gen.go b/storagemarket/types/legacytypes/network/types_cbor_gen.go new file mode 100644 index 000000000..e002fd641 --- /dev/null +++ b/storagemarket/types/legacytypes/network/types_cbor_gen.go @@ -0,0 +1,927 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package network + +import ( + "fmt" + "io" + "math" + "sort" + + legacytypes "github.com/filecoin-project/boost/storagemarket/types/legacytypes" + market "github.com/filecoin-project/go-state-types/builtin/v9/market" + crypto "github.com/filecoin-project/go-state-types/crypto" + cid "github.com/ipfs/go-cid" + cbg "github.com/whyrusleeping/cbor-gen" + xerrors "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +func (t *AskRequest) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{161}); err != nil { + return err + } + + // t.Miner (address.Address) (struct) + if len("Miner") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Miner\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Miner"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Miner")); err != nil { + return err + } + + if err := t.Miner.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *AskRequest) UnmarshalCBOR(r io.Reader) (err error) { + *t = AskRequest{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("AskRequest: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Miner (address.Address) (struct) + case "Miner": + + { + + if err := t.Miner.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Miner: %w", err) + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *AskResponse) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{161}); err != nil { + return err + } + + // t.Ask (legacytypes.SignedStorageAsk) (struct) + if len("Ask") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Ask\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Ask"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Ask")); err != nil { + return err + } + + if err := t.Ask.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *AskResponse) UnmarshalCBOR(r io.Reader) (err error) { + *t = AskResponse{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("AskResponse: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Ask (legacytypes.SignedStorageAsk) (struct) + case "Ask": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Ask = new(legacytypes.SignedStorageAsk) + if err := t.Ask.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Ask pointer: %w", err) + } + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *Proposal) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{163}); err != nil { + return err + } + + // t.Piece (legacytypes.DataRef) (struct) + if len("Piece") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Piece\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Piece"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Piece")); err != nil { + return err + } + + if err := t.Piece.MarshalCBOR(cw); err != nil { + return err + } + + // t.DealProposal (market.ClientDealProposal) (struct) + if len("DealProposal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DealProposal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DealProposal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DealProposal")); err != nil { + return err + } + + if err := t.DealProposal.MarshalCBOR(cw); err != nil { + return err + } + + // t.FastRetrieval (bool) (bool) + if len("FastRetrieval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"FastRetrieval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("FastRetrieval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("FastRetrieval")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.FastRetrieval); err != nil { + return err + } + return nil +} + +func (t *Proposal) UnmarshalCBOR(r io.Reader) (err error) { + *t = Proposal{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("Proposal: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Piece (legacytypes.DataRef) (struct) + case "Piece": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Piece = new(legacytypes.DataRef) + if err := t.Piece.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Piece pointer: %w", err) + } + } + + } + // t.DealProposal (market.ClientDealProposal) (struct) + case "DealProposal": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.DealProposal = new(market.ClientDealProposal) + if err := t.DealProposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DealProposal pointer: %w", err) + } + } + + } + // t.FastRetrieval (bool) (bool) + case "FastRetrieval": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.FastRetrieval = false + case 21: + t.FastRetrieval = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *Response) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{164}); err != nil { + return err + } + + // t.State (uint64) (uint64) + if len("State") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"State\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("State"))); err != nil { + return err + } + if _, err := cw.WriteString(string("State")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.State)); err != nil { + return err + } + + // t.Message (string) (string) + if len("Message") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Message\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Message"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Message")); err != nil { + return err + } + + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.Proposal (cid.Cid) (struct) + if len("Proposal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Proposal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Proposal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Proposal")); err != nil { + return err + } + + if err := cbg.WriteCid(cw, t.Proposal); err != nil { + return xerrors.Errorf("failed to write cid field t.Proposal: %w", err) + } + + // t.PublishMessage (cid.Cid) (struct) + if len("PublishMessage") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PublishMessage\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PublishMessage"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PublishMessage")); err != nil { + return err + } + + if t.PublishMessage == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PublishMessage); err != nil { + return xerrors.Errorf("failed to write cid field t.PublishMessage: %w", err) + } + } + + return nil +} + +func (t *Response) UnmarshalCBOR(r io.Reader) (err error) { + *t = Response{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("Response: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.State (uint64) (uint64) + case "State": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.State = uint64(extra) + + } + // t.Message (string) (string) + case "Message": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.Proposal (cid.Cid) (struct) + case "Proposal": + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.Proposal: %w", err) + } + + t.Proposal = c + + } + // t.PublishMessage (cid.Cid) (struct) + case "PublishMessage": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PublishMessage: %w", err) + } + + t.PublishMessage = &c + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *SignedResponse) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.Response (network.Response) (struct) + if len("Response") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Response\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Response"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Response")); err != nil { + return err + } + + if err := t.Response.MarshalCBOR(cw); err != nil { + return err + } + + // t.Signature (crypto.Signature) (struct) + if len("Signature") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Signature\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Signature"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Signature")); err != nil { + return err + } + + if err := t.Signature.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *SignedResponse) UnmarshalCBOR(r io.Reader) (err error) { + *t = SignedResponse{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("SignedResponse: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Response (network.Response) (struct) + case "Response": + + { + + if err := t.Response.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Response: %w", err) + } + + } + // t.Signature (crypto.Signature) (struct) + case "Signature": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Signature = new(crypto.Signature) + if err := t.Signature.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Signature pointer: %w", err) + } + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *DealStatusRequest) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.Proposal (cid.Cid) (struct) + if len("Proposal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Proposal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Proposal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Proposal")); err != nil { + return err + } + + if err := cbg.WriteCid(cw, t.Proposal); err != nil { + return xerrors.Errorf("failed to write cid field t.Proposal: %w", err) + } + + // t.Signature (crypto.Signature) (struct) + if len("Signature") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Signature\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Signature"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Signature")); err != nil { + return err + } + + if err := t.Signature.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *DealStatusRequest) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealStatusRequest{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("DealStatusRequest: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Proposal (cid.Cid) (struct) + case "Proposal": + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.Proposal: %w", err) + } + + t.Proposal = c + + } + // t.Signature (crypto.Signature) (struct) + case "Signature": + + { + + if err := t.Signature.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Signature: %w", err) + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *DealStatusResponse) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.DealState (legacytypes.ProviderDealState) (struct) + if len("DealState") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DealState\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DealState"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DealState")); err != nil { + return err + } + + if err := t.DealState.MarshalCBOR(cw); err != nil { + return err + } + + // t.Signature (crypto.Signature) (struct) + if len("Signature") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Signature\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Signature"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Signature")); err != nil { + return err + } + + if err := t.Signature.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *DealStatusResponse) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealStatusResponse{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("DealStatusResponse: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.DealState (legacytypes.ProviderDealState) (struct) + case "DealState": + + { + + if err := t.DealState.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DealState: %w", err) + } + + } + // t.Signature (crypto.Signature) (struct) + case "Signature": + + { + + if err := t.Signature.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Signature: %w", err) + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} diff --git a/storagemarket/types/legacytypes/types.go b/storagemarket/types/legacytypes/types.go new file mode 100644 index 000000000..440a898c7 --- /dev/null +++ b/storagemarket/types/legacytypes/types.go @@ -0,0 +1,319 @@ +package legacytypes + +import ( + "fmt" + "time" + + "github.com/filecoin-project/boost/datatransfer" + "github.com/ipfs/go-cid" + logging "github.com/ipfs/go-log/v2" + "github.com/libp2p/go-libp2p/core/peer" + ma "github.com/multiformats/go-multiaddr" + cbg "github.com/whyrusleeping/cbor-gen" + + "github.com/filecoin-project/go-address" + "github.com/filecoin-project/go-state-types/abi" + "github.com/filecoin-project/go-state-types/builtin/v9/market" + "github.com/filecoin-project/go-state-types/crypto" + + "github.com/filecoin-project/boost/storagemarket/types/legacytypes/filestore" +) + +var log = logging.Logger("storagemrkt") + +//go:generate cbor-gen-for --map-encoding ClientDeal MinerDeal Balance SignedStorageAsk StorageAsk DataRef ProviderDealState DealStages DealStage Log + +// The ID for the libp2p protocol for proposing storage deals. +const DealProtocolID101 = "/fil/storage/mk/1.0.1" +const DealProtocolID110 = "/fil/storage/mk/1.1.0" +const DealProtocolID111 = "/fil/storage/mk/1.1.1" + +// AskProtocolID is the ID for the libp2p protocol for querying miners for their current StorageAsk. +const OldAskProtocolID = "/fil/storage/ask/1.0.1" +const AskProtocolID = "/fil/storage/ask/1.1.0" + +// DealStatusProtocolID is the ID for the libp2p protocol for querying miners for the current status of a deal. +const OldDealStatusProtocolID = "/fil/storage/status/1.0.1" +const DealStatusProtocolID = "/fil/storage/status/1.1.0" + +// Balance represents a current balance of funds in the StorageMarketActor. +type Balance struct { + Locked abi.TokenAmount + Available abi.TokenAmount +} + +// StorageAsk defines the parameters by which a miner will choose to accept or +// reject a deal. Note: making a storage deal proposal which matches the miner's +// ask is a precondition, but not sufficient to ensure the deal is accepted (the +// storage provider may run its own decision logic). +type StorageAsk struct { + // Price per GiB / Epoch + Price abi.TokenAmount + VerifiedPrice abi.TokenAmount + + MinPieceSize abi.PaddedPieceSize + MaxPieceSize abi.PaddedPieceSize + Miner address.Address + Timestamp abi.ChainEpoch + Expiry abi.ChainEpoch + SeqNo uint64 +} + +// SignedStorageAsk is an ask signed by the miner's private key +type SignedStorageAsk struct { + Ask *StorageAsk + Signature *crypto.Signature +} + +// SignedStorageAskUndefined represents the empty value for SignedStorageAsk +var SignedStorageAskUndefined = SignedStorageAsk{} + +// StorageAskOption allows custom configuration of a storage ask +type StorageAskOption func(*StorageAsk) + +// MinPieceSize configures a minimum piece size of a StorageAsk +func MinPieceSize(minPieceSize abi.PaddedPieceSize) StorageAskOption { + return func(sa *StorageAsk) { + sa.MinPieceSize = minPieceSize + } +} + +// MaxPieceSize configures maxiumum piece size of a StorageAsk +func MaxPieceSize(maxPieceSize abi.PaddedPieceSize) StorageAskOption { + return func(sa *StorageAsk) { + sa.MaxPieceSize = maxPieceSize + } +} + +// StorageAskUndefined represents an empty value for StorageAsk +var StorageAskUndefined = StorageAsk{} + +type ClientDealProposal = market.ClientDealProposal + +// MinerDeal is the local state tracked for a deal by a StorageProvider +type MinerDeal struct { + ClientDealProposal + ProposalCid cid.Cid + AddFundsCid *cid.Cid + PublishCid *cid.Cid + Miner peer.ID + Client peer.ID + State StorageDealStatus + PiecePath filestore.Path + MetadataPath filestore.Path + SlashEpoch abi.ChainEpoch + FastRetrieval bool + Message string + FundsReserved abi.TokenAmount + Ref *DataRef + AvailableForRetrieval bool + + DealID abi.DealID + CreationTime cbg.CborTime + + TransferChannelId *datatransfer.ChannelID + SectorNumber abi.SectorNumber + + InboundCAR string +} + +// NewDealStages creates a new DealStages object ready to be used. +// EXPERIMENTAL; subject to change. +func NewDealStages() *DealStages { + return &DealStages{} +} + +// DealStages captures a timeline of the progress of a deal, grouped by stages. +// EXPERIMENTAL; subject to change. +type DealStages struct { + // Stages contains an entry for every stage that the deal has gone through. + // Each stage then contains logs. + Stages []*DealStage +} + +// DealStages captures data about the execution of a deal stage. +// EXPERIMENTAL; subject to change. +type DealStage struct { + // Human-readable fields. + // TODO: these _will_ need to be converted to canonical representations, so + // they are machine readable. + Name string + Description string + ExpectedDuration string + + // Timestamps. + // TODO: may be worth adding an exit timestamp. It _could_ be inferred from + // the start of the next stage, or from the timestamp of the last log line + // if this is a terminal stage. But that's non-determistic and it relies on + // assumptions. + CreatedTime cbg.CborTime + UpdatedTime cbg.CborTime + + // Logs contains a detailed timeline of events that occurred inside + // this stage. + Logs []*Log +} + +// Log represents a point-in-time event that occurred inside a deal stage. +// EXPERIMENTAL; subject to change. +type Log struct { + // Log is a human readable message. + // + // TODO: this _may_ need to be converted to a canonical data model so it + // is machine-readable. + Log string + + UpdatedTime cbg.CborTime +} + +// GetStage returns the DealStage object for a named stage, or nil if not found. +// +// TODO: the input should be a strongly-typed enum instead of a free-form string. +// TODO: drop Get from GetStage to make this code more idiomatic. Return a +// second ok boolean to make it even more idiomatic. +// EXPERIMENTAL; subject to change. +func (ds *DealStages) GetStage(stage string) *DealStage { + if ds == nil { + return nil + } + + for _, s := range ds.Stages { + if s.Name == stage { + return s + } + } + + return nil +} + +// AddStageLog adds a log to the specified stage, creating the stage if it +// doesn't exist yet. +// EXPERIMENTAL; subject to change. +func (ds *DealStages) AddStageLog(stage, description, expectedDuration, msg string) { + if ds == nil { + return + } + + log.Debugf("adding log for stage <%s> msg <%s>", stage, msg) + + now := curTime() + st := ds.GetStage(stage) + if st == nil { + st = &DealStage{ + CreatedTime: now, + } + ds.Stages = append(ds.Stages, st) + } + + st.Name = stage + st.Description = description + st.ExpectedDuration = expectedDuration + st.UpdatedTime = now + if msg != "" && (len(st.Logs) == 0 || st.Logs[len(st.Logs)-1].Log != msg) { + // only add the log if it's not a duplicate. + st.Logs = append(st.Logs, &Log{msg, now}) + } +} + +// AddLog adds a log inside the DealStages object of the deal. +// EXPERIMENTAL; subject to change. +func (d *ClientDeal) AddLog(msg string, a ...interface{}) { + if len(a) > 0 { + msg = fmt.Sprintf(msg, a...) + } + + stage := DealStates[d.State] + description := DealStatesDescriptions[d.State] + expectedDuration := DealStatesDurations[d.State] + + d.DealStages.AddStageLog(stage, description, expectedDuration, msg) +} + +// ClientDeal is the local state tracked for a deal by a StorageClient +type ClientDeal struct { + market.ClientDealProposal + ProposalCid cid.Cid + AddFundsCid *cid.Cid + State StorageDealStatus + Miner peer.ID + MinerWorker address.Address + DealID abi.DealID + DataRef *DataRef + Message string + DealStages *DealStages + PublishMessage *cid.Cid + SlashEpoch abi.ChainEpoch + PollRetryCount uint64 + PollErrorCount uint64 + FastRetrieval bool + FundsReserved abi.TokenAmount + CreationTime cbg.CborTime + TransferChannelID *datatransfer.ChannelID + SectorNumber abi.SectorNumber +} + +// StorageProviderInfo describes on chain information about a StorageProvider +// (use QueryAsk to determine more specific deal parameters) +type StorageProviderInfo struct { + Address address.Address // actor address + Owner address.Address + Worker address.Address // signs messages + SectorSize uint64 + PeerID peer.ID + Addrs []ma.Multiaddr +} + +// ProposeStorageDealResult returns the result for a proposing a deal +type ProposeStorageDealResult struct { + ProposalCid cid.Cid +} + +// ProposeStorageDealParams describes the parameters for proposing a storage deal +type ProposeStorageDealParams struct { + Addr address.Address + Info *StorageProviderInfo + Data *DataRef + StartEpoch abi.ChainEpoch + EndEpoch abi.ChainEpoch + Price abi.TokenAmount + Collateral abi.TokenAmount + Rt abi.RegisteredSealProof + FastRetrieval bool + VerifiedDeal bool +} + +const ( + // TTGraphsync means data for a deal will be transferred by graphsync + TTGraphsync = "graphsync" + + // TTManual means data for a deal will be transferred manually and imported + // on the provider + TTManual = "manual" +) + +// DataRef is a reference for how data will be transferred for a given storage deal +type DataRef struct { + TransferType string + Root cid.Cid + + PieceCid *cid.Cid // Optional for non-manual transfer, will be recomputed from the data if not given + PieceSize abi.UnpaddedPieceSize // Optional for non-manual transfer, will be recomputed from the data if not given + RawBlockSize uint64 // Optional: used as the denominator when calculating transfer % +} + +// ProviderDealState represents a Provider's current state of a deal +type ProviderDealState struct { + State StorageDealStatus + Message string + Proposal *market.DealProposal + ProposalCid *cid.Cid + AddFundsCid *cid.Cid + PublishCid *cid.Cid + DealID abi.DealID + FastRetrieval bool +} + +func curTime() cbg.CborTime { + now := time.Now() + return cbg.CborTime(time.Unix(0, now.UnixNano()).UTC()) +} diff --git a/storagemarket/types/legacytypes/types_cbor_gen.go b/storagemarket/types/legacytypes/types_cbor_gen.go new file mode 100644 index 000000000..f3cbe405b --- /dev/null +++ b/storagemarket/types/legacytypes/types_cbor_gen.go @@ -0,0 +1,3180 @@ +// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT. + +package legacytypes + +import ( + "fmt" + "io" + "math" + "sort" + + datatransfer "github.com/filecoin-project/boost/datatransfer" + filestore "github.com/filecoin-project/boost/storagemarket/types/legacytypes/filestore" + abi "github.com/filecoin-project/go-state-types/abi" + market "github.com/filecoin-project/go-state-types/builtin/v9/market" + crypto "github.com/filecoin-project/go-state-types/crypto" + cid "github.com/ipfs/go-cid" + peer "github.com/libp2p/go-libp2p/core/peer" + cbg "github.com/whyrusleeping/cbor-gen" + xerrors "golang.org/x/xerrors" +) + +var _ = xerrors.Errorf +var _ = cid.Undef +var _ = math.E +var _ = sort.Sort + +func (t *ClientDeal) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{179}); err != nil { + return err + } + + // t.Miner (peer.ID) (string) + if len("Miner") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Miner\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Miner"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Miner")); err != nil { + return err + } + + if len(t.Miner) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Miner was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Miner))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Miner)); err != nil { + return err + } + + // t.State (uint64) (uint64) + if len("State") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"State\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("State"))); err != nil { + return err + } + if _, err := cw.WriteString(string("State")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.State)); err != nil { + return err + } + + // t.DealID (abi.DealID) (uint64) + if len("DealID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DealID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DealID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DealID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.DealID)); err != nil { + return err + } + + // t.DataRef (legacytypes.DataRef) (struct) + if len("DataRef") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DataRef\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DataRef"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DataRef")); err != nil { + return err + } + + if err := t.DataRef.MarshalCBOR(cw); err != nil { + return err + } + + // t.Message (string) (string) + if len("Message") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Message\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Message"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Message")); err != nil { + return err + } + + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.DealStages (legacytypes.DealStages) (struct) + if len("DealStages") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DealStages\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DealStages"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DealStages")); err != nil { + return err + } + + if err := t.DealStages.MarshalCBOR(cw); err != nil { + return err + } + + // t.SlashEpoch (abi.ChainEpoch) (int64) + if len("SlashEpoch") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"SlashEpoch\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("SlashEpoch"))); err != nil { + return err + } + if _, err := cw.WriteString(string("SlashEpoch")); err != nil { + return err + } + + if t.SlashEpoch >= 0 { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.SlashEpoch)); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajNegativeInt, uint64(-t.SlashEpoch-1)); err != nil { + return err + } + } + + // t.AddFundsCid (cid.Cid) (struct) + if len("AddFundsCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"AddFundsCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("AddFundsCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("AddFundsCid")); err != nil { + return err + } + + if t.AddFundsCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.AddFundsCid); err != nil { + return xerrors.Errorf("failed to write cid field t.AddFundsCid: %w", err) + } + } + + // t.MinerWorker (address.Address) (struct) + if len("MinerWorker") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"MinerWorker\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("MinerWorker"))); err != nil { + return err + } + if _, err := cw.WriteString(string("MinerWorker")); err != nil { + return err + } + + if err := t.MinerWorker.MarshalCBOR(cw); err != nil { + return err + } + + // t.ProposalCid (cid.Cid) (struct) + if len("ProposalCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ProposalCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ProposalCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ProposalCid")); err != nil { + return err + } + + if err := cbg.WriteCid(cw, t.ProposalCid); err != nil { + return xerrors.Errorf("failed to write cid field t.ProposalCid: %w", err) + } + + // t.CreationTime (typegen.CborTime) (struct) + if len("CreationTime") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"CreationTime\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("CreationTime"))); err != nil { + return err + } + if _, err := cw.WriteString(string("CreationTime")); err != nil { + return err + } + + if err := t.CreationTime.MarshalCBOR(cw); err != nil { + return err + } + + // t.SectorNumber (abi.SectorNumber) (uint64) + if len("SectorNumber") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"SectorNumber\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("SectorNumber"))); err != nil { + return err + } + if _, err := cw.WriteString(string("SectorNumber")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.SectorNumber)); err != nil { + return err + } + + // t.FastRetrieval (bool) (bool) + if len("FastRetrieval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"FastRetrieval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("FastRetrieval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("FastRetrieval")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.FastRetrieval); err != nil { + return err + } + + // t.FundsReserved (big.Int) (struct) + if len("FundsReserved") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"FundsReserved\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("FundsReserved"))); err != nil { + return err + } + if _, err := cw.WriteString(string("FundsReserved")); err != nil { + return err + } + + if err := t.FundsReserved.MarshalCBOR(cw); err != nil { + return err + } + + // t.PollErrorCount (uint64) (uint64) + if len("PollErrorCount") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PollErrorCount\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PollErrorCount"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PollErrorCount")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PollErrorCount)); err != nil { + return err + } + + // t.PollRetryCount (uint64) (uint64) + if len("PollRetryCount") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PollRetryCount\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PollRetryCount"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PollRetryCount")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PollRetryCount)); err != nil { + return err + } + + // t.PublishMessage (cid.Cid) (struct) + if len("PublishMessage") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PublishMessage\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PublishMessage"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PublishMessage")); err != nil { + return err + } + + if t.PublishMessage == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PublishMessage); err != nil { + return xerrors.Errorf("failed to write cid field t.PublishMessage: %w", err) + } + } + + // t.TransferChannelID (datatransfer.ChannelID) (struct) + if len("TransferChannelID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"TransferChannelID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("TransferChannelID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("TransferChannelID")); err != nil { + return err + } + + if err := t.TransferChannelID.MarshalCBOR(cw); err != nil { + return err + } + + // t.ClientDealProposal (market.ClientDealProposal) (struct) + if len("ClientDealProposal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ClientDealProposal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ClientDealProposal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ClientDealProposal")); err != nil { + return err + } + + if err := t.ClientDealProposal.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *ClientDeal) UnmarshalCBOR(r io.Reader) (err error) { + *t = ClientDeal{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("ClientDeal: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Miner (peer.ID) (string) + case "Miner": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Miner = peer.ID(sval) + } + // t.State (uint64) (uint64) + case "State": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.State = uint64(extra) + + } + // t.DealID (abi.DealID) (uint64) + case "DealID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.DealID = abi.DealID(extra) + + } + // t.DataRef (legacytypes.DataRef) (struct) + case "DataRef": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.DataRef = new(DataRef) + if err := t.DataRef.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DataRef pointer: %w", err) + } + } + + } + // t.Message (string) (string) + case "Message": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.DealStages (legacytypes.DealStages) (struct) + case "DealStages": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.DealStages = new(DealStages) + if err := t.DealStages.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.DealStages pointer: %w", err) + } + } + + } + // t.SlashEpoch (abi.ChainEpoch) (int64) + case "SlashEpoch": + { + maj, extra, err := cr.ReadHeader() + var extraI int64 + if err != nil { + return err + } + switch maj { + case cbg.MajUnsignedInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 positive overflow") + } + case cbg.MajNegativeInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 negative overflow") + } + extraI = -1 - extraI + default: + return fmt.Errorf("wrong type for int64 field: %d", maj) + } + + t.SlashEpoch = abi.ChainEpoch(extraI) + } + // t.AddFundsCid (cid.Cid) (struct) + case "AddFundsCid": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.AddFundsCid: %w", err) + } + + t.AddFundsCid = &c + } + + } + // t.MinerWorker (address.Address) (struct) + case "MinerWorker": + + { + + if err := t.MinerWorker.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.MinerWorker: %w", err) + } + + } + // t.ProposalCid (cid.Cid) (struct) + case "ProposalCid": + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.ProposalCid: %w", err) + } + + t.ProposalCid = c + + } + // t.CreationTime (typegen.CborTime) (struct) + case "CreationTime": + + { + + if err := t.CreationTime.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.CreationTime: %w", err) + } + + } + // t.SectorNumber (abi.SectorNumber) (uint64) + case "SectorNumber": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.SectorNumber = abi.SectorNumber(extra) + + } + // t.FastRetrieval (bool) (bool) + case "FastRetrieval": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.FastRetrieval = false + case 21: + t.FastRetrieval = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.FundsReserved (big.Int) (struct) + case "FundsReserved": + + { + + if err := t.FundsReserved.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.FundsReserved: %w", err) + } + + } + // t.PollErrorCount (uint64) (uint64) + case "PollErrorCount": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PollErrorCount = uint64(extra) + + } + // t.PollRetryCount (uint64) (uint64) + case "PollRetryCount": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PollRetryCount = uint64(extra) + + } + // t.PublishMessage (cid.Cid) (struct) + case "PublishMessage": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PublishMessage: %w", err) + } + + t.PublishMessage = &c + } + + } + // t.TransferChannelID (datatransfer.ChannelID) (struct) + case "TransferChannelID": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.TransferChannelID = new(datatransfer.ChannelID) + if err := t.TransferChannelID.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.TransferChannelID pointer: %w", err) + } + } + + } + // t.ClientDealProposal (market.ClientDealProposal) (struct) + case "ClientDealProposal": + + { + + if err := t.ClientDealProposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ClientDealProposal: %w", err) + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *MinerDeal) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{180}); err != nil { + return err + } + + // t.Ref (legacytypes.DataRef) (struct) + if len("Ref") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Ref\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Ref"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Ref")); err != nil { + return err + } + + if err := t.Ref.MarshalCBOR(cw); err != nil { + return err + } + + // t.Miner (peer.ID) (string) + if len("Miner") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Miner\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Miner"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Miner")); err != nil { + return err + } + + if len(t.Miner) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Miner was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Miner))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Miner)); err != nil { + return err + } + + // t.State (uint64) (uint64) + if len("State") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"State\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("State"))); err != nil { + return err + } + if _, err := cw.WriteString(string("State")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.State)); err != nil { + return err + } + + // t.Client (peer.ID) (string) + if len("Client") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Client\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Client"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Client")); err != nil { + return err + } + + if len(t.Client) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Client was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Client))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Client)); err != nil { + return err + } + + // t.DealID (abi.DealID) (uint64) + if len("DealID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DealID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DealID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DealID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.DealID)); err != nil { + return err + } + + // t.Message (string) (string) + if len("Message") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Message\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Message"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Message")); err != nil { + return err + } + + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.PiecePath (filestore.Path) (string) + if len("PiecePath") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PiecePath\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PiecePath"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PiecePath")); err != nil { + return err + } + + if len(t.PiecePath) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.PiecePath was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.PiecePath))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.PiecePath)); err != nil { + return err + } + + // t.InboundCAR (string) (string) + if len("InboundCAR") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"InboundCAR\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("InboundCAR"))); err != nil { + return err + } + if _, err := cw.WriteString(string("InboundCAR")); err != nil { + return err + } + + if len(t.InboundCAR) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.InboundCAR was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.InboundCAR))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.InboundCAR)); err != nil { + return err + } + + // t.PublishCid (cid.Cid) (struct) + if len("PublishCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PublishCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PublishCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PublishCid")); err != nil { + return err + } + + if t.PublishCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PublishCid); err != nil { + return xerrors.Errorf("failed to write cid field t.PublishCid: %w", err) + } + } + + // t.SlashEpoch (abi.ChainEpoch) (int64) + if len("SlashEpoch") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"SlashEpoch\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("SlashEpoch"))); err != nil { + return err + } + if _, err := cw.WriteString(string("SlashEpoch")); err != nil { + return err + } + + if t.SlashEpoch >= 0 { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.SlashEpoch)); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajNegativeInt, uint64(-t.SlashEpoch-1)); err != nil { + return err + } + } + + // t.AddFundsCid (cid.Cid) (struct) + if len("AddFundsCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"AddFundsCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("AddFundsCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("AddFundsCid")); err != nil { + return err + } + + if t.AddFundsCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.AddFundsCid); err != nil { + return xerrors.Errorf("failed to write cid field t.AddFundsCid: %w", err) + } + } + + // t.ProposalCid (cid.Cid) (struct) + if len("ProposalCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ProposalCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ProposalCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ProposalCid")); err != nil { + return err + } + + if err := cbg.WriteCid(cw, t.ProposalCid); err != nil { + return xerrors.Errorf("failed to write cid field t.ProposalCid: %w", err) + } + + // t.CreationTime (typegen.CborTime) (struct) + if len("CreationTime") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"CreationTime\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("CreationTime"))); err != nil { + return err + } + if _, err := cw.WriteString(string("CreationTime")); err != nil { + return err + } + + if err := t.CreationTime.MarshalCBOR(cw); err != nil { + return err + } + + // t.MetadataPath (filestore.Path) (string) + if len("MetadataPath") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"MetadataPath\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("MetadataPath"))); err != nil { + return err + } + if _, err := cw.WriteString(string("MetadataPath")); err != nil { + return err + } + + if len(t.MetadataPath) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.MetadataPath was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.MetadataPath))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.MetadataPath)); err != nil { + return err + } + + // t.SectorNumber (abi.SectorNumber) (uint64) + if len("SectorNumber") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"SectorNumber\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("SectorNumber"))); err != nil { + return err + } + if _, err := cw.WriteString(string("SectorNumber")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.SectorNumber)); err != nil { + return err + } + + // t.FastRetrieval (bool) (bool) + if len("FastRetrieval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"FastRetrieval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("FastRetrieval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("FastRetrieval")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.FastRetrieval); err != nil { + return err + } + + // t.FundsReserved (big.Int) (struct) + if len("FundsReserved") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"FundsReserved\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("FundsReserved"))); err != nil { + return err + } + if _, err := cw.WriteString(string("FundsReserved")); err != nil { + return err + } + + if err := t.FundsReserved.MarshalCBOR(cw); err != nil { + return err + } + + // t.TransferChannelId (datatransfer.ChannelID) (struct) + if len("TransferChannelId") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"TransferChannelId\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("TransferChannelId"))); err != nil { + return err + } + if _, err := cw.WriteString(string("TransferChannelId")); err != nil { + return err + } + + if err := t.TransferChannelId.MarshalCBOR(cw); err != nil { + return err + } + + // t.ClientDealProposal (market.ClientDealProposal) (struct) + if len("ClientDealProposal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ClientDealProposal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ClientDealProposal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ClientDealProposal")); err != nil { + return err + } + + if err := t.ClientDealProposal.MarshalCBOR(cw); err != nil { + return err + } + + // t.AvailableForRetrieval (bool) (bool) + if len("AvailableForRetrieval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"AvailableForRetrieval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("AvailableForRetrieval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("AvailableForRetrieval")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.AvailableForRetrieval); err != nil { + return err + } + return nil +} + +func (t *MinerDeal) UnmarshalCBOR(r io.Reader) (err error) { + *t = MinerDeal{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("MinerDeal: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Ref (legacytypes.DataRef) (struct) + case "Ref": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Ref = new(DataRef) + if err := t.Ref.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Ref pointer: %w", err) + } + } + + } + // t.Miner (peer.ID) (string) + case "Miner": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Miner = peer.ID(sval) + } + // t.State (uint64) (uint64) + case "State": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.State = uint64(extra) + + } + // t.Client (peer.ID) (string) + case "Client": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Client = peer.ID(sval) + } + // t.DealID (abi.DealID) (uint64) + case "DealID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.DealID = abi.DealID(extra) + + } + // t.Message (string) (string) + case "Message": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.PiecePath (filestore.Path) (string) + case "PiecePath": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.PiecePath = filestore.Path(sval) + } + // t.InboundCAR (string) (string) + case "InboundCAR": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.InboundCAR = string(sval) + } + // t.PublishCid (cid.Cid) (struct) + case "PublishCid": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PublishCid: %w", err) + } + + t.PublishCid = &c + } + + } + // t.SlashEpoch (abi.ChainEpoch) (int64) + case "SlashEpoch": + { + maj, extra, err := cr.ReadHeader() + var extraI int64 + if err != nil { + return err + } + switch maj { + case cbg.MajUnsignedInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 positive overflow") + } + case cbg.MajNegativeInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 negative overflow") + } + extraI = -1 - extraI + default: + return fmt.Errorf("wrong type for int64 field: %d", maj) + } + + t.SlashEpoch = abi.ChainEpoch(extraI) + } + // t.AddFundsCid (cid.Cid) (struct) + case "AddFundsCid": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.AddFundsCid: %w", err) + } + + t.AddFundsCid = &c + } + + } + // t.ProposalCid (cid.Cid) (struct) + case "ProposalCid": + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.ProposalCid: %w", err) + } + + t.ProposalCid = c + + } + // t.CreationTime (typegen.CborTime) (struct) + case "CreationTime": + + { + + if err := t.CreationTime.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.CreationTime: %w", err) + } + + } + // t.MetadataPath (filestore.Path) (string) + case "MetadataPath": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.MetadataPath = filestore.Path(sval) + } + // t.SectorNumber (abi.SectorNumber) (uint64) + case "SectorNumber": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.SectorNumber = abi.SectorNumber(extra) + + } + // t.FastRetrieval (bool) (bool) + case "FastRetrieval": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.FastRetrieval = false + case 21: + t.FastRetrieval = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + // t.FundsReserved (big.Int) (struct) + case "FundsReserved": + + { + + if err := t.FundsReserved.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.FundsReserved: %w", err) + } + + } + // t.TransferChannelId (datatransfer.ChannelID) (struct) + case "TransferChannelId": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.TransferChannelId = new(datatransfer.ChannelID) + if err := t.TransferChannelId.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.TransferChannelId pointer: %w", err) + } + } + + } + // t.ClientDealProposal (market.ClientDealProposal) (struct) + case "ClientDealProposal": + + { + + if err := t.ClientDealProposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.ClientDealProposal: %w", err) + } + + } + // t.AvailableForRetrieval (bool) (bool) + case "AvailableForRetrieval": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.AvailableForRetrieval = false + case 21: + t.AvailableForRetrieval = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *Balance) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.Locked (big.Int) (struct) + if len("Locked") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Locked\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Locked"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Locked")); err != nil { + return err + } + + if err := t.Locked.MarshalCBOR(cw); err != nil { + return err + } + + // t.Available (big.Int) (struct) + if len("Available") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Available\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Available"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Available")); err != nil { + return err + } + + if err := t.Available.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *Balance) UnmarshalCBOR(r io.Reader) (err error) { + *t = Balance{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("Balance: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Locked (big.Int) (struct) + case "Locked": + + { + + if err := t.Locked.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Locked: %w", err) + } + + } + // t.Available (big.Int) (struct) + case "Available": + + { + + if err := t.Available.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Available: %w", err) + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *SignedStorageAsk) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.Ask (legacytypes.StorageAsk) (struct) + if len("Ask") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Ask\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Ask"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Ask")); err != nil { + return err + } + + if err := t.Ask.MarshalCBOR(cw); err != nil { + return err + } + + // t.Signature (crypto.Signature) (struct) + if len("Signature") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Signature\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Signature"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Signature")); err != nil { + return err + } + + if err := t.Signature.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *SignedStorageAsk) UnmarshalCBOR(r io.Reader) (err error) { + *t = SignedStorageAsk{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("SignedStorageAsk: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Ask (legacytypes.StorageAsk) (struct) + case "Ask": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Ask = new(StorageAsk) + if err := t.Ask.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Ask pointer: %w", err) + } + } + + } + // t.Signature (crypto.Signature) (struct) + case "Signature": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Signature = new(crypto.Signature) + if err := t.Signature.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Signature pointer: %w", err) + } + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *StorageAsk) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{168}); err != nil { + return err + } + + // t.Miner (address.Address) (struct) + if len("Miner") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Miner\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Miner"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Miner")); err != nil { + return err + } + + if err := t.Miner.MarshalCBOR(cw); err != nil { + return err + } + + // t.Price (big.Int) (struct) + if len("Price") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Price\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Price"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Price")); err != nil { + return err + } + + if err := t.Price.MarshalCBOR(cw); err != nil { + return err + } + + // t.SeqNo (uint64) (uint64) + if len("SeqNo") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"SeqNo\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("SeqNo"))); err != nil { + return err + } + if _, err := cw.WriteString(string("SeqNo")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.SeqNo)); err != nil { + return err + } + + // t.Expiry (abi.ChainEpoch) (int64) + if len("Expiry") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Expiry\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Expiry"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Expiry")); err != nil { + return err + } + + if t.Expiry >= 0 { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Expiry)); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajNegativeInt, uint64(-t.Expiry-1)); err != nil { + return err + } + } + + // t.Timestamp (abi.ChainEpoch) (int64) + if len("Timestamp") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Timestamp\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Timestamp"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Timestamp")); err != nil { + return err + } + + if t.Timestamp >= 0 { + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.Timestamp)); err != nil { + return err + } + } else { + if err := cw.WriteMajorTypeHeader(cbg.MajNegativeInt, uint64(-t.Timestamp-1)); err != nil { + return err + } + } + + // t.MaxPieceSize (abi.PaddedPieceSize) (uint64) + if len("MaxPieceSize") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"MaxPieceSize\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("MaxPieceSize"))); err != nil { + return err + } + if _, err := cw.WriteString(string("MaxPieceSize")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.MaxPieceSize)); err != nil { + return err + } + + // t.MinPieceSize (abi.PaddedPieceSize) (uint64) + if len("MinPieceSize") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"MinPieceSize\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("MinPieceSize"))); err != nil { + return err + } + if _, err := cw.WriteString(string("MinPieceSize")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.MinPieceSize)); err != nil { + return err + } + + // t.VerifiedPrice (big.Int) (struct) + if len("VerifiedPrice") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"VerifiedPrice\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("VerifiedPrice"))); err != nil { + return err + } + if _, err := cw.WriteString(string("VerifiedPrice")); err != nil { + return err + } + + if err := t.VerifiedPrice.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *StorageAsk) UnmarshalCBOR(r io.Reader) (err error) { + *t = StorageAsk{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("StorageAsk: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Miner (address.Address) (struct) + case "Miner": + + { + + if err := t.Miner.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Miner: %w", err) + } + + } + // t.Price (big.Int) (struct) + case "Price": + + { + + if err := t.Price.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Price: %w", err) + } + + } + // t.SeqNo (uint64) (uint64) + case "SeqNo": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.SeqNo = uint64(extra) + + } + // t.Expiry (abi.ChainEpoch) (int64) + case "Expiry": + { + maj, extra, err := cr.ReadHeader() + var extraI int64 + if err != nil { + return err + } + switch maj { + case cbg.MajUnsignedInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 positive overflow") + } + case cbg.MajNegativeInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 negative overflow") + } + extraI = -1 - extraI + default: + return fmt.Errorf("wrong type for int64 field: %d", maj) + } + + t.Expiry = abi.ChainEpoch(extraI) + } + // t.Timestamp (abi.ChainEpoch) (int64) + case "Timestamp": + { + maj, extra, err := cr.ReadHeader() + var extraI int64 + if err != nil { + return err + } + switch maj { + case cbg.MajUnsignedInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 positive overflow") + } + case cbg.MajNegativeInt: + extraI = int64(extra) + if extraI < 0 { + return fmt.Errorf("int64 negative overflow") + } + extraI = -1 - extraI + default: + return fmt.Errorf("wrong type for int64 field: %d", maj) + } + + t.Timestamp = abi.ChainEpoch(extraI) + } + // t.MaxPieceSize (abi.PaddedPieceSize) (uint64) + case "MaxPieceSize": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.MaxPieceSize = abi.PaddedPieceSize(extra) + + } + // t.MinPieceSize (abi.PaddedPieceSize) (uint64) + case "MinPieceSize": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.MinPieceSize = abi.PaddedPieceSize(extra) + + } + // t.VerifiedPrice (big.Int) (struct) + case "VerifiedPrice": + + { + + if err := t.VerifiedPrice.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.VerifiedPrice: %w", err) + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *DataRef) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{165}); err != nil { + return err + } + + // t.Root (cid.Cid) (struct) + if len("Root") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Root\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Root"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Root")); err != nil { + return err + } + + if err := cbg.WriteCid(cw, t.Root); err != nil { + return xerrors.Errorf("failed to write cid field t.Root: %w", err) + } + + // t.PieceCid (cid.Cid) (struct) + if len("PieceCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PieceCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PieceCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PieceCid")); err != nil { + return err + } + + if t.PieceCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PieceCid); err != nil { + return xerrors.Errorf("failed to write cid field t.PieceCid: %w", err) + } + } + + // t.PieceSize (abi.UnpaddedPieceSize) (uint64) + if len("PieceSize") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PieceSize\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PieceSize"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PieceSize")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.PieceSize)); err != nil { + return err + } + + // t.RawBlockSize (uint64) (uint64) + if len("RawBlockSize") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"RawBlockSize\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("RawBlockSize"))); err != nil { + return err + } + if _, err := cw.WriteString(string("RawBlockSize")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.RawBlockSize)); err != nil { + return err + } + + // t.TransferType (string) (string) + if len("TransferType") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"TransferType\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("TransferType"))); err != nil { + return err + } + if _, err := cw.WriteString(string("TransferType")); err != nil { + return err + } + + if len(t.TransferType) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.TransferType was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.TransferType))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.TransferType)); err != nil { + return err + } + return nil +} + +func (t *DataRef) UnmarshalCBOR(r io.Reader) (err error) { + *t = DataRef{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("DataRef: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Root (cid.Cid) (struct) + case "Root": + + { + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.Root: %w", err) + } + + t.Root = c + + } + // t.PieceCid (cid.Cid) (struct) + case "PieceCid": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PieceCid: %w", err) + } + + t.PieceCid = &c + } + + } + // t.PieceSize (abi.UnpaddedPieceSize) (uint64) + case "PieceSize": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.PieceSize = abi.UnpaddedPieceSize(extra) + + } + // t.RawBlockSize (uint64) (uint64) + case "RawBlockSize": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.RawBlockSize = uint64(extra) + + } + // t.TransferType (string) (string) + case "TransferType": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.TransferType = string(sval) + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *ProviderDealState) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{168}); err != nil { + return err + } + + // t.State (uint64) (uint64) + if len("State") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"State\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("State"))); err != nil { + return err + } + if _, err := cw.WriteString(string("State")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.State)); err != nil { + return err + } + + // t.DealID (abi.DealID) (uint64) + if len("DealID") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"DealID\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("DealID"))); err != nil { + return err + } + if _, err := cw.WriteString(string("DealID")); err != nil { + return err + } + + if err := cw.WriteMajorTypeHeader(cbg.MajUnsignedInt, uint64(t.DealID)); err != nil { + return err + } + + // t.Message (string) (string) + if len("Message") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Message\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Message"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Message")); err != nil { + return err + } + + if len(t.Message) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Message was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Message))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Message)); err != nil { + return err + } + + // t.Proposal (market.DealProposal) (struct) + if len("Proposal") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Proposal\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Proposal"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Proposal")); err != nil { + return err + } + + if err := t.Proposal.MarshalCBOR(cw); err != nil { + return err + } + + // t.PublishCid (cid.Cid) (struct) + if len("PublishCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"PublishCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("PublishCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("PublishCid")); err != nil { + return err + } + + if t.PublishCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.PublishCid); err != nil { + return xerrors.Errorf("failed to write cid field t.PublishCid: %w", err) + } + } + + // t.AddFundsCid (cid.Cid) (struct) + if len("AddFundsCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"AddFundsCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("AddFundsCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("AddFundsCid")); err != nil { + return err + } + + if t.AddFundsCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.AddFundsCid); err != nil { + return xerrors.Errorf("failed to write cid field t.AddFundsCid: %w", err) + } + } + + // t.ProposalCid (cid.Cid) (struct) + if len("ProposalCid") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ProposalCid\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ProposalCid"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ProposalCid")); err != nil { + return err + } + + if t.ProposalCid == nil { + if _, err := cw.Write(cbg.CborNull); err != nil { + return err + } + } else { + if err := cbg.WriteCid(cw, *t.ProposalCid); err != nil { + return xerrors.Errorf("failed to write cid field t.ProposalCid: %w", err) + } + } + + // t.FastRetrieval (bool) (bool) + if len("FastRetrieval") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"FastRetrieval\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("FastRetrieval"))); err != nil { + return err + } + if _, err := cw.WriteString(string("FastRetrieval")); err != nil { + return err + } + + if err := cbg.WriteBool(w, t.FastRetrieval); err != nil { + return err + } + return nil +} + +func (t *ProviderDealState) UnmarshalCBOR(r io.Reader) (err error) { + *t = ProviderDealState{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("ProviderDealState: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.State (uint64) (uint64) + case "State": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.State = uint64(extra) + + } + // t.DealID (abi.DealID) (uint64) + case "DealID": + + { + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajUnsignedInt { + return fmt.Errorf("wrong type for uint64 field") + } + t.DealID = abi.DealID(extra) + + } + // t.Message (string) (string) + case "Message": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Message = string(sval) + } + // t.Proposal (market.DealProposal) (struct) + case "Proposal": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Proposal = new(market.DealProposal) + if err := t.Proposal.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Proposal pointer: %w", err) + } + } + + } + // t.PublishCid (cid.Cid) (struct) + case "PublishCid": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.PublishCid: %w", err) + } + + t.PublishCid = &c + } + + } + // t.AddFundsCid (cid.Cid) (struct) + case "AddFundsCid": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.AddFundsCid: %w", err) + } + + t.AddFundsCid = &c + } + + } + // t.ProposalCid (cid.Cid) (struct) + case "ProposalCid": + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + + c, err := cbg.ReadCid(cr) + if err != nil { + return xerrors.Errorf("failed to read cid field t.ProposalCid: %w", err) + } + + t.ProposalCid = &c + } + + } + // t.FastRetrieval (bool) (bool) + case "FastRetrieval": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + if maj != cbg.MajOther { + return fmt.Errorf("booleans must be major type 7") + } + switch extra { + case 20: + t.FastRetrieval = false + case 21: + t.FastRetrieval = true + default: + return fmt.Errorf("booleans are either major type 7, value 20 or 21 (got %d)", extra) + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *DealStages) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{161}); err != nil { + return err + } + + // t.Stages ([]*legacytypes.DealStage) (slice) + if len("Stages") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Stages\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Stages"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Stages")); err != nil { + return err + } + + if len(t.Stages) > cbg.MaxLength { + return xerrors.Errorf("Slice value in field t.Stages was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajArray, uint64(len(t.Stages))); err != nil { + return err + } + for _, v := range t.Stages { + if err := v.MarshalCBOR(cw); err != nil { + return err + } + + } + return nil +} + +func (t *DealStages) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealStages{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("DealStages: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Stages ([]*legacytypes.DealStage) (slice) + case "Stages": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + + if extra > cbg.MaxLength { + return fmt.Errorf("t.Stages: array too large (%d)", extra) + } + + if maj != cbg.MajArray { + return fmt.Errorf("expected cbor array") + } + + if extra > 0 { + t.Stages = make([]*DealStage, extra) + } + + for i := 0; i < int(extra); i++ { + { + var maj byte + var extra uint64 + var err error + _ = maj + _ = extra + _ = err + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Stages[i] = new(DealStage) + if err := t.Stages[i].UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Stages[i] pointer: %w", err) + } + } + + } + + } + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *DealStage) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{166}); err != nil { + return err + } + + // t.Logs ([]*legacytypes.Log) (slice) + if len("Logs") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Logs\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Logs"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Logs")); err != nil { + return err + } + + if len(t.Logs) > cbg.MaxLength { + return xerrors.Errorf("Slice value in field t.Logs was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajArray, uint64(len(t.Logs))); err != nil { + return err + } + for _, v := range t.Logs { + if err := v.MarshalCBOR(cw); err != nil { + return err + } + + } + + // t.Name (string) (string) + if len("Name") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Name\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Name"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Name")); err != nil { + return err + } + + if len(t.Name) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Name was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Name))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Name)); err != nil { + return err + } + + // t.CreatedTime (typegen.CborTime) (struct) + if len("CreatedTime") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"CreatedTime\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("CreatedTime"))); err != nil { + return err + } + if _, err := cw.WriteString(string("CreatedTime")); err != nil { + return err + } + + if err := t.CreatedTime.MarshalCBOR(cw); err != nil { + return err + } + + // t.Description (string) (string) + if len("Description") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Description\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Description"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Description")); err != nil { + return err + } + + if len(t.Description) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Description was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Description))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Description)); err != nil { + return err + } + + // t.UpdatedTime (typegen.CborTime) (struct) + if len("UpdatedTime") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"UpdatedTime\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("UpdatedTime"))); err != nil { + return err + } + if _, err := cw.WriteString(string("UpdatedTime")); err != nil { + return err + } + + if err := t.UpdatedTime.MarshalCBOR(cw); err != nil { + return err + } + + // t.ExpectedDuration (string) (string) + if len("ExpectedDuration") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"ExpectedDuration\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("ExpectedDuration"))); err != nil { + return err + } + if _, err := cw.WriteString(string("ExpectedDuration")); err != nil { + return err + } + + if len(t.ExpectedDuration) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.ExpectedDuration was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.ExpectedDuration))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.ExpectedDuration)); err != nil { + return err + } + return nil +} + +func (t *DealStage) UnmarshalCBOR(r io.Reader) (err error) { + *t = DealStage{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("DealStage: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Logs ([]*legacytypes.Log) (slice) + case "Logs": + + maj, extra, err = cr.ReadHeader() + if err != nil { + return err + } + + if extra > cbg.MaxLength { + return fmt.Errorf("t.Logs: array too large (%d)", extra) + } + + if maj != cbg.MajArray { + return fmt.Errorf("expected cbor array") + } + + if extra > 0 { + t.Logs = make([]*Log, extra) + } + + for i := 0; i < int(extra); i++ { + { + var maj byte + var extra uint64 + var err error + _ = maj + _ = extra + _ = err + + { + + b, err := cr.ReadByte() + if err != nil { + return err + } + if b != cbg.CborNull[0] { + if err := cr.UnreadByte(); err != nil { + return err + } + t.Logs[i] = new(Log) + if err := t.Logs[i].UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.Logs[i] pointer: %w", err) + } + } + + } + + } + } + // t.Name (string) (string) + case "Name": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Name = string(sval) + } + // t.CreatedTime (typegen.CborTime) (struct) + case "CreatedTime": + + { + + if err := t.CreatedTime.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.CreatedTime: %w", err) + } + + } + // t.Description (string) (string) + case "Description": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Description = string(sval) + } + // t.UpdatedTime (typegen.CborTime) (struct) + case "UpdatedTime": + + { + + if err := t.UpdatedTime.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.UpdatedTime: %w", err) + } + + } + // t.ExpectedDuration (string) (string) + case "ExpectedDuration": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.ExpectedDuration = string(sval) + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} +func (t *Log) MarshalCBOR(w io.Writer) error { + if t == nil { + _, err := w.Write(cbg.CborNull) + return err + } + + cw := cbg.NewCborWriter(w) + + if _, err := cw.Write([]byte{162}); err != nil { + return err + } + + // t.Log (string) (string) + if len("Log") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"Log\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("Log"))); err != nil { + return err + } + if _, err := cw.WriteString(string("Log")); err != nil { + return err + } + + if len(t.Log) > cbg.MaxLength { + return xerrors.Errorf("Value in field t.Log was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len(t.Log))); err != nil { + return err + } + if _, err := cw.WriteString(string(t.Log)); err != nil { + return err + } + + // t.UpdatedTime (typegen.CborTime) (struct) + if len("UpdatedTime") > cbg.MaxLength { + return xerrors.Errorf("Value in field \"UpdatedTime\" was too long") + } + + if err := cw.WriteMajorTypeHeader(cbg.MajTextString, uint64(len("UpdatedTime"))); err != nil { + return err + } + if _, err := cw.WriteString(string("UpdatedTime")); err != nil { + return err + } + + if err := t.UpdatedTime.MarshalCBOR(cw); err != nil { + return err + } + return nil +} + +func (t *Log) UnmarshalCBOR(r io.Reader) (err error) { + *t = Log{} + + cr := cbg.NewCborReader(r) + + maj, extra, err := cr.ReadHeader() + if err != nil { + return err + } + defer func() { + if err == io.EOF { + err = io.ErrUnexpectedEOF + } + }() + + if maj != cbg.MajMap { + return fmt.Errorf("cbor input should be of type map") + } + + if extra > cbg.MaxLength { + return fmt.Errorf("Log: map struct too large (%d)", extra) + } + + var name string + n := extra + + for i := uint64(0); i < n; i++ { + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + name = string(sval) + } + + switch name { + // t.Log (string) (string) + case "Log": + + { + sval, err := cbg.ReadString(cr) + if err != nil { + return err + } + + t.Log = string(sval) + } + // t.UpdatedTime (typegen.CborTime) (struct) + case "UpdatedTime": + + { + + if err := t.UpdatedTime.UnmarshalCBOR(cr); err != nil { + return xerrors.Errorf("unmarshaling t.UpdatedTime: %w", err) + } + + } + + default: + // Field doesn't exist on this type, so ignore it + cbg.ScanForLinks(r, func(cid.Cid) {}) + } + } + + return nil +} diff --git a/storagemarket/types/mock_types/mocks.go b/storagemarket/types/mock_types/mocks.go index bbd01124b..e15cd4810 100644 --- a/storagemarket/types/mock_types/mocks.go +++ b/storagemarket/types/mock_types/mocks.go @@ -9,7 +9,6 @@ import ( io "io" reflect "reflect" - storagemarket "github.com/filecoin-project/boost-gfm/storagemarket" types "github.com/filecoin-project/boost/storagemarket/types" abi "github.com/filecoin-project/go-state-types/abi" market "github.com/filecoin-project/go-state-types/builtin/v9/market" @@ -157,10 +156,10 @@ func (m *MockChainDealManager) EXPECT() *MockChainDealManagerMockRecorder { } // WaitForPublishDeals mocks base method. -func (m *MockChainDealManager) WaitForPublishDeals(arg0 context.Context, arg1 cid.Cid, arg2 market.DealProposal) (*storagemarket.PublishDealsWaitResult, error) { +func (m *MockChainDealManager) WaitForPublishDeals(arg0 context.Context, arg1 cid.Cid, arg2 market.DealProposal) (*types.PublishDealsWaitResult, error) { m.ctrl.T.Helper() ret := m.ctrl.Call(m, "WaitForPublishDeals", arg0, arg1, arg2) - ret0, _ := ret[0].(*storagemarket.PublishDealsWaitResult) + ret0, _ := ret[0].(*types.PublishDealsWaitResult) ret1, _ := ret[1].(error) return ret0, ret1 } diff --git a/storagemarket/types/types.go b/storagemarket/types/types.go index d7390c320..98bc0da47 100644 --- a/storagemarket/types/types.go +++ b/storagemarket/types/types.go @@ -7,7 +7,7 @@ import ( "io" "net/url" - "github.com/filecoin-project/boost-gfm/storagemarket" + "github.com/filecoin-project/boost/storagemarket/types/legacytypes" "github.com/filecoin-project/boost/transport/httptransport/util" "github.com/filecoin-project/boost/transport/types" "github.com/filecoin-project/go-address" @@ -168,7 +168,7 @@ type DealPublisher interface { } type ChainDealManager interface { - WaitForPublishDeals(ctx context.Context, publishCid cid.Cid, proposal market.DealProposal) (*storagemarket.PublishDealsWaitResult, error) + WaitForPublishDeals(ctx context.Context, publishCid cid.Cid, proposal market.DealProposal) (*PublishDealsWaitResult, error) } type IndexProvider interface { @@ -178,9 +178,16 @@ type IndexProvider interface { } type AskGetter interface { - GetAsk() *storagemarket.SignedStorageAsk + GetAsk(miner address.Address) *legacytypes.SignedStorageAsk } type SignatureVerifier interface { VerifySignature(ctx context.Context, sig crypto.Signature, addr address.Address, input []byte) (bool, error) } + +// PublishDealsWaitResult is the result of a call to wait for publish deals to +// appear on chain +type PublishDealsWaitResult struct { + DealID abi.DealID + FinalCid cid.Cid +}