From be99dc552826eeacee3b6797ffdc6343f942b632 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Michael=20Mur=C3=A9?= Date: Wed, 27 Jul 2022 17:08:28 +0200 Subject: [PATCH 1/2] blockstore: allow to pass a file to write in This allow the caller to control the file lifecycle and implement a cleaning strategy. An example would be: - open a file in a temporary folder - if the write suceed, close the file and move it to its final destination - on error, remove the file If the underlying filesystem support it, an anonymous file can be used instead: open then immediately delete, use the file descriptor to write. A crash before the end of the write process would release the used storage automatically. --- v2/blockstore/readwrite.go | 16 ++++++++++++++-- 1 file changed, 14 insertions(+), 2 deletions(-) diff --git a/v2/blockstore/readwrite.go b/v2/blockstore/readwrite.go index 774f37ff..f29c206f 100644 --- a/v2/blockstore/readwrite.go +++ b/v2/blockstore/readwrite.go @@ -10,12 +10,13 @@ import ( blocks "github.com/ipfs/go-block-format" "github.com/ipfs/go-cid" blockstore "github.com/ipfs/go-ipfs-blockstore" + "github.com/multiformats/go-varint" + carv2 "github.com/ipld/go-car/v2" "github.com/ipld/go-car/v2/index" "github.com/ipld/go-car/v2/internal/carv1" "github.com/ipld/go-car/v2/internal/carv1/util" internalio "github.com/ipld/go-car/v2/internal/io" - "github.com/multiformats/go-varint" ) var _ blockstore.Blockstore = (*ReadWrite)(nil) @@ -103,6 +104,18 @@ func OpenReadWrite(path string, roots []cid.Cid, opts ...carv2.Option) (*ReadWri if err != nil { return nil, fmt.Errorf("could not open read/write file: %w", err) } + rwbs, err := OpenReadWriteFile(f, roots, opts...) + if err != nil { + return nil, err + } + // close the file when finalizing + rwbs.ronly.carv2Closer = rwbs.f + return rwbs, nil +} + +// OpenReadWriteFile is similar as OpenReadWrite but let you control the file lifecycle. +// You are responsible to close the given file. +func OpenReadWriteFile(f *os.File, roots []cid.Cid, opts ...carv2.Option) (*ReadWrite, error) { 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 @@ -145,7 +158,6 @@ func OpenReadWrite(path string, roots []cid.Cid, opts ...carv2.Option) (*ReadWri } rwbs.ronly.backing = v1r rwbs.ronly.idx = rwbs.idx - rwbs.ronly.carv2Closer = rwbs.f if resume { if err = rwbs.resumeWithRoots(!rwbs.opts.WriteAsCarV1, roots); err != nil { From aac3995eb5aaf11061abe25b6f72f50f91a6e7d1 Mon Sep 17 00:00:00 2001 From: Rod Vagg Date: Mon, 1 Aug 2022 17:29:47 +1000 Subject: [PATCH 2/2] Update v2/blockstore/readwrite.go --- v2/blockstore/readwrite.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/v2/blockstore/readwrite.go b/v2/blockstore/readwrite.go index f29c206f..542d0df4 100644 --- a/v2/blockstore/readwrite.go +++ b/v2/blockstore/readwrite.go @@ -113,8 +113,8 @@ func OpenReadWrite(path string, roots []cid.Cid, opts ...carv2.Option) (*ReadWri return rwbs, nil } -// OpenReadWriteFile is similar as OpenReadWrite but let you control the file lifecycle. -// You are responsible to close the given file. +// OpenReadWriteFile is similar as OpenReadWrite but lets you control the file lifecycle. +// You are responsible for closing the given file. func OpenReadWriteFile(f *os.File, roots []cid.Cid, opts ...carv2.Option) (*ReadWrite, error) { stat, err := f.Stat() if err != nil {