From c3989d190609e6ad667dd5d7c3b1693bf376629b Mon Sep 17 00:00:00 2001 From: Edwin Mackenzie-Owen Date: Tue, 25 Jul 2023 09:31:36 +0200 Subject: [PATCH] smb: implement multi-threaded writes for copies to smb smb2.File implements the WriterAtCloser interface defined in fs/types.go. Expose it via a OpenWriterAt method on the fs struct to support multi-threaded writes. --- backend/smb/smb.go | 39 +++++++++++++++++++++++++++++++++++++++ 1 file changed, 39 insertions(+) diff --git a/backend/smb/smb.go b/backend/smb/smb.go index 6f1640cf4..b9ad5353b 100644 --- a/backend/smb/smb.go +++ b/backend/smb/smb.go @@ -475,6 +475,45 @@ func (f *Fs) About(ctx context.Context) (_ *fs.Usage, err error) { return usage, nil } +// OpenWriterAt opens with a handle for random access writes +// +// Pass in the remote desired and the size if known. +// +// It truncates any existing object +func (f *Fs) OpenWriterAt(ctx context.Context, remote string, size int64) (fs.WriterAtCloser, error) { + var err error + o := &Object{ + fs: f, + remote: remote, + } + share, filename := o.split() + if share == "" || filename == "" { + return nil, fs.ErrorIsDir + } + + err = o.fs.ensureDirectory(ctx, share, filename) + if err != nil { + return nil, fmt.Errorf("failed to make parent directories: %w", err) + } + + filename = o.fs.toSambaPath(filename) + + o.fs.addSession() // Show session in use + defer o.fs.removeSession() + + cn, err := o.fs.getConnection(ctx, share) + if err != nil { + return nil, err + } + + fl, err := cn.smbShare.OpenFile(filename, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0o644) + if err != nil { + return nil, fmt.Errorf("failed to open: %w", err) + } + + return fl, nil +} + // Shutdown the backend, closing any background tasks and any // cached connections. func (f *Fs) Shutdown(ctx context.Context) error {