mirror of
https://github.com/wavetermdev/waveterm
synced 2026-05-21 15:58:30 +00:00
This adds the ability to stream `tar` archives over channels between `wsh` instances. The main use cases for this are remote copy and move operations. It also completes the `wavefs` implementation of the FileShare interface to allow copy/move interoperability between wavefiles and other storage types. The tar streaming functionality has been broken out into the new `tarcopy` package for easy reuse. New `fileshare` functions are added for `CopyInternal`, which allows copying files internal to a filesystem to bypass the expensive interop layer, and `MoveInternal`, which does the same for moving a file within a filesystem. Copying between remotes is now handled by `CopyRemote`, which accepts the source `FileShareClient` as a parameter. `wsh` connections use the same implementation for `CopyInternal` and `CopyRemote` as they need to request the channel on the remote destination, since we don't offer a way to pass channels as a parameter to a remote call. This also adds a recursive `-r` flag to `wsh file rm` to allow for deleting a directory and all its contents. S3 support will be addressed in a future PR. --------- Co-authored-by: sawka <mike@commandline.dev> Co-authored-by: Copilot Autofix powered by AI <62310815+github-advanced-security[bot]@users.noreply.github.com> Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com>
586 lines
18 KiB
Go
586 lines
18 KiB
Go
// Copyright 2025, Command Line Inc.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package wavefs
|
|
|
|
import (
|
|
"archive/tar"
|
|
"context"
|
|
"encoding/base64"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"io/fs"
|
|
"log"
|
|
"path"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/wavetermdev/waveterm/pkg/filestore"
|
|
"github.com/wavetermdev/waveterm/pkg/remote/connparse"
|
|
"github.com/wavetermdev/waveterm/pkg/remote/fileshare/fstype"
|
|
"github.com/wavetermdev/waveterm/pkg/util/fileutil"
|
|
"github.com/wavetermdev/waveterm/pkg/util/iochan/iochantypes"
|
|
"github.com/wavetermdev/waveterm/pkg/util/tarcopy"
|
|
"github.com/wavetermdev/waveterm/pkg/util/wavefileutil"
|
|
"github.com/wavetermdev/waveterm/pkg/waveobj"
|
|
"github.com/wavetermdev/waveterm/pkg/wps"
|
|
"github.com/wavetermdev/waveterm/pkg/wshrpc"
|
|
"github.com/wavetermdev/waveterm/pkg/wshutil"
|
|
)
|
|
|
|
const (
|
|
DefaultTimeout = 30 * time.Second
|
|
)
|
|
|
|
type WaveClient struct{}
|
|
|
|
var _ fstype.FileShareClient = WaveClient{}
|
|
|
|
func NewWaveClient() *WaveClient {
|
|
return &WaveClient{}
|
|
}
|
|
|
|
func (c WaveClient) ReadStream(ctx context.Context, conn *connparse.Connection, data wshrpc.FileData) <-chan wshrpc.RespOrErrorUnion[wshrpc.FileData] {
|
|
ch := make(chan wshrpc.RespOrErrorUnion[wshrpc.FileData], 16)
|
|
go func() {
|
|
defer close(ch)
|
|
rtnData, err := c.Read(ctx, conn, data)
|
|
if err != nil {
|
|
ch <- wshutil.RespErr[wshrpc.FileData](err)
|
|
return
|
|
}
|
|
dataLen := len(rtnData.Data64)
|
|
if !rtnData.Info.IsDir {
|
|
for i := 0; i < dataLen; i += wshrpc.FileChunkSize {
|
|
if ctx.Err() != nil {
|
|
ch <- wshutil.RespErr[wshrpc.FileData](ctx.Err())
|
|
return
|
|
}
|
|
dataEnd := min(i+wshrpc.FileChunkSize, dataLen)
|
|
ch <- wshrpc.RespOrErrorUnion[wshrpc.FileData]{Response: wshrpc.FileData{Data64: rtnData.Data64[i:dataEnd], Info: rtnData.Info, At: &wshrpc.FileDataAt{Offset: int64(i), Size: dataEnd - i}}}
|
|
}
|
|
} else {
|
|
for i := 0; i < len(rtnData.Entries); i += wshrpc.DirChunkSize {
|
|
if ctx.Err() != nil {
|
|
ch <- wshutil.RespErr[wshrpc.FileData](ctx.Err())
|
|
return
|
|
}
|
|
ch <- wshrpc.RespOrErrorUnion[wshrpc.FileData]{Response: wshrpc.FileData{Entries: rtnData.Entries[i:min(i+wshrpc.DirChunkSize, len(rtnData.Entries))], Info: rtnData.Info}}
|
|
}
|
|
}
|
|
}()
|
|
return ch
|
|
}
|
|
|
|
func (c WaveClient) Read(ctx context.Context, conn *connparse.Connection, data wshrpc.FileData) (*wshrpc.FileData, error) {
|
|
zoneId := conn.Host
|
|
if zoneId == "" {
|
|
return nil, fmt.Errorf("zoneid not found in connection")
|
|
}
|
|
fileName, err := cleanPath(conn.Path)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error cleaning path: %w", err)
|
|
}
|
|
if data.At != nil {
|
|
_, dataBuf, err := filestore.WFS.ReadAt(ctx, zoneId, fileName, data.At.Offset, int64(data.At.Size))
|
|
if err == nil {
|
|
return &wshrpc.FileData{Info: data.Info, Data64: base64.StdEncoding.EncodeToString(dataBuf)}, nil
|
|
} else if errors.Is(err, fs.ErrNotExist) {
|
|
return nil, fmt.Errorf("NOTFOUND: %w", err)
|
|
} else {
|
|
return nil, fmt.Errorf("error reading blockfile: %w", err)
|
|
}
|
|
} else {
|
|
_, dataBuf, err := filestore.WFS.ReadFile(ctx, zoneId, fileName)
|
|
if err == nil {
|
|
return &wshrpc.FileData{Info: data.Info, Data64: base64.StdEncoding.EncodeToString(dataBuf)}, nil
|
|
} else if !errors.Is(err, fs.ErrNotExist) {
|
|
return nil, fmt.Errorf("error reading blockfile: %w", err)
|
|
}
|
|
}
|
|
list, err := c.ListEntries(ctx, conn, nil)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error listing blockfiles: %w", err)
|
|
}
|
|
return &wshrpc.FileData{Info: data.Info, Entries: list}, nil
|
|
}
|
|
|
|
func (c WaveClient) ReadTarStream(ctx context.Context, conn *connparse.Connection, opts *wshrpc.FileCopyOpts) <-chan wshrpc.RespOrErrorUnion[iochantypes.Packet] {
|
|
log.Printf("ReadTarStream: conn: %v, opts: %v\n", conn, opts)
|
|
list, err := c.ListEntries(ctx, conn, nil)
|
|
if err != nil {
|
|
return wshutil.SendErrCh[iochantypes.Packet](fmt.Errorf("error listing blockfiles: %w", err))
|
|
}
|
|
|
|
pathPrefix := getPathPrefix(conn)
|
|
schemeAndHost := conn.GetSchemeAndHost() + "/"
|
|
|
|
timeout := DefaultTimeout
|
|
if opts.Timeout > 0 {
|
|
timeout = time.Duration(opts.Timeout) * time.Millisecond
|
|
}
|
|
readerCtx, cancel := context.WithTimeout(context.Background(), timeout)
|
|
rtn, writeHeader, fileWriter, tarClose := tarcopy.TarCopySrc(readerCtx, pathPrefix)
|
|
|
|
go func() {
|
|
defer func() {
|
|
tarClose()
|
|
cancel()
|
|
}()
|
|
for _, file := range list {
|
|
if readerCtx.Err() != nil {
|
|
rtn <- wshutil.RespErr[iochantypes.Packet](readerCtx.Err())
|
|
return
|
|
}
|
|
file.Mode = 0644
|
|
|
|
if err = writeHeader(fileutil.ToFsFileInfo(file), file.Path); err != nil {
|
|
rtn <- wshutil.RespErr[iochantypes.Packet](fmt.Errorf("error writing tar header: %w", err))
|
|
return
|
|
}
|
|
if file.IsDir {
|
|
continue
|
|
}
|
|
|
|
log.Printf("ReadTarStream: reading file: %s\n", file.Path)
|
|
|
|
internalPath := strings.TrimPrefix(file.Path, schemeAndHost)
|
|
|
|
_, dataBuf, err := filestore.WFS.ReadFile(ctx, conn.Host, internalPath)
|
|
if err != nil {
|
|
rtn <- wshutil.RespErr[iochantypes.Packet](fmt.Errorf("error reading blockfile: %w", err))
|
|
return
|
|
}
|
|
if _, err = fileWriter.Write(dataBuf); err != nil {
|
|
rtn <- wshutil.RespErr[iochantypes.Packet](fmt.Errorf("error writing tar data: %w", err))
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
|
|
return rtn
|
|
}
|
|
|
|
func (c WaveClient) ListEntriesStream(ctx context.Context, conn *connparse.Connection, opts *wshrpc.FileListOpts) <-chan wshrpc.RespOrErrorUnion[wshrpc.CommandRemoteListEntriesRtnData] {
|
|
ch := make(chan wshrpc.RespOrErrorUnion[wshrpc.CommandRemoteListEntriesRtnData], 16)
|
|
go func() {
|
|
defer close(ch)
|
|
list, err := c.ListEntries(ctx, conn, opts)
|
|
if err != nil {
|
|
ch <- wshutil.RespErr[wshrpc.CommandRemoteListEntriesRtnData](err)
|
|
return
|
|
}
|
|
for i := 0; i < len(list); i += wshrpc.DirChunkSize {
|
|
ch <- wshrpc.RespOrErrorUnion[wshrpc.CommandRemoteListEntriesRtnData]{Response: wshrpc.CommandRemoteListEntriesRtnData{FileInfo: list[i:min(i+wshrpc.DirChunkSize, len(list))]}}
|
|
}
|
|
}()
|
|
return ch
|
|
}
|
|
|
|
func (c WaveClient) ListEntries(ctx context.Context, conn *connparse.Connection, opts *wshrpc.FileListOpts) ([]*wshrpc.FileInfo, error) {
|
|
log.Printf("ListEntries: conn: %v, opts: %v\n", conn, opts)
|
|
zoneId := conn.Host
|
|
if zoneId == "" {
|
|
return nil, fmt.Errorf("zoneid not found in connection")
|
|
}
|
|
if opts == nil {
|
|
opts = &wshrpc.FileListOpts{}
|
|
}
|
|
prefix, err := cleanPath(conn.Path)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error cleaning path: %w", err)
|
|
}
|
|
fileListOrig, err := filestore.WFS.ListFiles(ctx, zoneId)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error listing blockfiles: %w", err)
|
|
}
|
|
var fileList []*wshrpc.FileInfo
|
|
for _, wf := range fileListOrig {
|
|
fileList = append(fileList, wavefileutil.WaveFileToFileInfo(wf))
|
|
}
|
|
if prefix != "" {
|
|
var filteredList []*wshrpc.FileInfo
|
|
for _, file := range fileList {
|
|
if strings.HasPrefix(file.Name, prefix) {
|
|
filteredList = append(filteredList, file)
|
|
}
|
|
}
|
|
fileList = filteredList
|
|
}
|
|
if !opts.All {
|
|
var filteredList []*wshrpc.FileInfo
|
|
dirMap := make(map[string]any) // the value is max modtime
|
|
for _, file := range fileList {
|
|
// if there is an extra "/" after the prefix, don't include it
|
|
// first strip the prefix
|
|
relPath := strings.TrimPrefix(file.Name, prefix)
|
|
// then check if there is a "/" after the prefix
|
|
if strings.Contains(relPath, "/") {
|
|
dirPath := strings.Split(relPath, "/")[0]
|
|
dirMap[dirPath] = struct{}{}
|
|
continue
|
|
}
|
|
filteredList = append(filteredList, file)
|
|
}
|
|
for dir := range dirMap {
|
|
dirName := prefix + dir + "/"
|
|
filteredList = append(filteredList, &wshrpc.FileInfo{
|
|
Path: fmt.Sprintf(wavefileutil.WaveFilePathPattern, zoneId, dirName),
|
|
Name: dirName,
|
|
Dir: dirName,
|
|
Size: 0,
|
|
IsDir: true,
|
|
SupportsMkdir: false,
|
|
})
|
|
}
|
|
fileList = filteredList
|
|
}
|
|
if opts.Offset > 0 {
|
|
if opts.Offset >= len(fileList) {
|
|
fileList = nil
|
|
} else {
|
|
fileList = fileList[opts.Offset:]
|
|
}
|
|
}
|
|
if opts.Limit > 0 {
|
|
if opts.Limit < len(fileList) {
|
|
fileList = fileList[:opts.Limit]
|
|
}
|
|
}
|
|
return fileList, nil
|
|
}
|
|
|
|
func (c WaveClient) Stat(ctx context.Context, conn *connparse.Connection) (*wshrpc.FileInfo, error) {
|
|
zoneId := conn.Host
|
|
if zoneId == "" {
|
|
return nil, fmt.Errorf("zoneid not found in connection")
|
|
}
|
|
fileName, err := cleanPath(conn.Path)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error cleaning path: %w", err)
|
|
}
|
|
fileInfo, err := filestore.WFS.Stat(ctx, zoneId, fileName)
|
|
if err != nil {
|
|
if errors.Is(err, fs.ErrNotExist) {
|
|
return nil, fmt.Errorf("NOTFOUND: %w", err)
|
|
}
|
|
return nil, fmt.Errorf("error getting file info: %w", err)
|
|
}
|
|
return wavefileutil.WaveFileToFileInfo(fileInfo), nil
|
|
}
|
|
|
|
func (c WaveClient) PutFile(ctx context.Context, conn *connparse.Connection, data wshrpc.FileData) error {
|
|
dataBuf, err := base64.StdEncoding.DecodeString(data.Data64)
|
|
if err != nil {
|
|
return fmt.Errorf("error decoding data64: %w", err)
|
|
}
|
|
zoneId := conn.Host
|
|
if zoneId == "" {
|
|
return fmt.Errorf("zoneid not found in connection")
|
|
}
|
|
fileName, err := cleanPath(conn.Path)
|
|
if err != nil {
|
|
return fmt.Errorf("error cleaning path: %w", err)
|
|
}
|
|
_, err = filestore.WFS.Stat(ctx, zoneId, fileName)
|
|
if err != nil {
|
|
if !errors.Is(err, fs.ErrNotExist) {
|
|
return fmt.Errorf("error getting blockfile info: %w", err)
|
|
}
|
|
var opts wshrpc.FileOpts
|
|
var meta wshrpc.FileMeta
|
|
if data.Info != nil {
|
|
if data.Info.Opts != nil {
|
|
opts = *data.Info.Opts
|
|
}
|
|
if data.Info.Meta != nil {
|
|
meta = *data.Info.Meta
|
|
}
|
|
}
|
|
err := filestore.WFS.MakeFile(ctx, zoneId, fileName, meta, opts)
|
|
if err != nil {
|
|
return fmt.Errorf("error making blockfile: %w", err)
|
|
}
|
|
}
|
|
if data.At != nil && data.At.Offset >= 0 {
|
|
err = filestore.WFS.WriteAt(ctx, zoneId, fileName, data.At.Offset, dataBuf)
|
|
if errors.Is(err, fs.ErrNotExist) {
|
|
return fmt.Errorf("NOTFOUND: %w", err)
|
|
}
|
|
if err != nil {
|
|
return fmt.Errorf("error writing to blockfile: %w", err)
|
|
}
|
|
} else {
|
|
err = filestore.WFS.WriteFile(ctx, zoneId, fileName, dataBuf)
|
|
if errors.Is(err, fs.ErrNotExist) {
|
|
return fmt.Errorf("NOTFOUND: %w", err)
|
|
}
|
|
if err != nil {
|
|
return fmt.Errorf("error writing to blockfile: %w", err)
|
|
}
|
|
}
|
|
wps.Broker.Publish(wps.WaveEvent{
|
|
Event: wps.Event_BlockFile,
|
|
Scopes: []string{waveobj.MakeORef(waveobj.OType_Block, zoneId).String()},
|
|
Data: &wps.WSFileEventData{
|
|
ZoneId: zoneId,
|
|
FileName: fileName,
|
|
FileOp: wps.FileOp_Invalidate,
|
|
},
|
|
})
|
|
return nil
|
|
}
|
|
|
|
func (c WaveClient) AppendFile(ctx context.Context, conn *connparse.Connection, data wshrpc.FileData) error {
|
|
dataBuf, err := base64.StdEncoding.DecodeString(data.Data64)
|
|
if err != nil {
|
|
return fmt.Errorf("error decoding data64: %w", err)
|
|
}
|
|
zoneId := conn.Host
|
|
if zoneId == "" {
|
|
return fmt.Errorf("zoneid not found in connection")
|
|
}
|
|
fileName, err := cleanPath(conn.Path)
|
|
if err != nil {
|
|
return fmt.Errorf("error cleaning path: %w", err)
|
|
}
|
|
_, err = filestore.WFS.Stat(ctx, zoneId, fileName)
|
|
if err != nil {
|
|
if !errors.Is(err, fs.ErrNotExist) {
|
|
return fmt.Errorf("error getting blockfile info: %w", err)
|
|
}
|
|
var opts wshrpc.FileOpts
|
|
var meta wshrpc.FileMeta
|
|
if data.Info != nil {
|
|
if data.Info.Opts != nil {
|
|
opts = *data.Info.Opts
|
|
}
|
|
if data.Info.Meta != nil {
|
|
meta = *data.Info.Meta
|
|
}
|
|
}
|
|
err := filestore.WFS.MakeFile(ctx, zoneId, fileName, meta, opts)
|
|
if err != nil {
|
|
return fmt.Errorf("error making blockfile: %w", err)
|
|
}
|
|
}
|
|
err = filestore.WFS.AppendData(ctx, zoneId, fileName, dataBuf)
|
|
if errors.Is(err, fs.ErrNotExist) {
|
|
return fmt.Errorf("NOTFOUND: %w", err)
|
|
}
|
|
if err != nil {
|
|
return fmt.Errorf("error writing to blockfile: %w", err)
|
|
}
|
|
wps.Broker.Publish(wps.WaveEvent{
|
|
Event: wps.Event_BlockFile,
|
|
Scopes: []string{waveobj.MakeORef(waveobj.OType_Block, zoneId).String()},
|
|
Data: &wps.WSFileEventData{
|
|
ZoneId: zoneId,
|
|
FileName: fileName,
|
|
FileOp: wps.FileOp_Invalidate,
|
|
},
|
|
})
|
|
return nil
|
|
}
|
|
|
|
// WaveFile does not support directories, only prefix-based listing
|
|
func (c WaveClient) Mkdir(ctx context.Context, conn *connparse.Connection) error {
|
|
return errors.ErrUnsupported
|
|
}
|
|
|
|
func (c WaveClient) MoveInternal(ctx context.Context, srcConn, destConn *connparse.Connection, opts *wshrpc.FileCopyOpts) error {
|
|
if srcConn.Host != destConn.Host {
|
|
return fmt.Errorf("move internal, src and dest hosts do not match")
|
|
}
|
|
err := c.CopyInternal(ctx, srcConn, destConn, opts)
|
|
if err != nil {
|
|
return fmt.Errorf("error copying blockfile: %w", err)
|
|
}
|
|
err = c.Delete(ctx, srcConn, opts.Recursive)
|
|
if err != nil {
|
|
return fmt.Errorf("error deleting blockfile: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c WaveClient) CopyInternal(ctx context.Context, srcConn, destConn *connparse.Connection, opts *wshrpc.FileCopyOpts) error {
|
|
if srcConn.Host == destConn.Host {
|
|
host := srcConn.Host
|
|
srcFileName, err := cleanPath(srcConn.Path)
|
|
if err != nil {
|
|
return fmt.Errorf("error cleaning source path: %w", err)
|
|
}
|
|
destFileName, err := cleanPath(destConn.Path)
|
|
if err != nil {
|
|
return fmt.Errorf("error cleaning destination path: %w", err)
|
|
}
|
|
err = filestore.WFS.MakeFile(ctx, host, destFileName, wshrpc.FileMeta{}, wshrpc.FileOpts{})
|
|
if err != nil {
|
|
return fmt.Errorf("error making source blockfile: %w", err)
|
|
}
|
|
_, dataBuf, err := filestore.WFS.ReadFile(ctx, host, srcFileName)
|
|
if err != nil {
|
|
return fmt.Errorf("error reading source blockfile: %w", err)
|
|
}
|
|
err = filestore.WFS.WriteFile(ctx, host, destFileName, dataBuf)
|
|
if err != nil {
|
|
return fmt.Errorf("error writing to destination blockfile: %w", err)
|
|
}
|
|
wps.Broker.Publish(wps.WaveEvent{
|
|
Event: wps.Event_BlockFile,
|
|
Scopes: []string{waveobj.MakeORef(waveobj.OType_Block, host).String()},
|
|
Data: &wps.WSFileEventData{
|
|
ZoneId: host,
|
|
FileName: destFileName,
|
|
FileOp: wps.FileOp_Invalidate,
|
|
},
|
|
})
|
|
return nil
|
|
} else {
|
|
return fmt.Errorf("copy between different hosts not supported")
|
|
}
|
|
}
|
|
|
|
func (c WaveClient) CopyRemote(ctx context.Context, srcConn, destConn *connparse.Connection, srcClient fstype.FileShareClient, opts *wshrpc.FileCopyOpts) error {
|
|
zoneId := destConn.Host
|
|
if zoneId == "" {
|
|
return fmt.Errorf("zoneid not found in connection")
|
|
}
|
|
destPrefix := getPathPrefix(destConn)
|
|
destPrefix = strings.TrimPrefix(destPrefix, destConn.GetSchemeAndHost()+"/")
|
|
log.Printf("CopyRemote: srcConn: %v, destConn: %v, destPrefix: %s\n", srcConn, destConn, destPrefix)
|
|
readCtx, cancel := context.WithCancelCause(ctx)
|
|
ioch := srcClient.ReadTarStream(readCtx, srcConn, opts)
|
|
err := tarcopy.TarCopyDest(readCtx, cancel, ioch, func(next *tar.Header, reader *tar.Reader) error {
|
|
if next.Typeflag == tar.TypeDir {
|
|
return nil
|
|
}
|
|
fileName, err := cleanPath(path.Join(destPrefix, next.Name))
|
|
if err != nil {
|
|
return fmt.Errorf("error cleaning path: %w", err)
|
|
}
|
|
_, err = filestore.WFS.Stat(ctx, zoneId, fileName)
|
|
if err != nil {
|
|
if !errors.Is(err, fs.ErrNotExist) {
|
|
return fmt.Errorf("error getting blockfile info: %w", err)
|
|
}
|
|
err := filestore.WFS.MakeFile(ctx, zoneId, fileName, nil, wshrpc.FileOpts{})
|
|
if err != nil {
|
|
return fmt.Errorf("error making blockfile: %w", err)
|
|
}
|
|
}
|
|
log.Printf("CopyRemote: writing file: %s; size: %d\n", fileName, next.Size)
|
|
dataBuf := make([]byte, next.Size)
|
|
_, err = reader.Read(dataBuf)
|
|
if err != nil {
|
|
if !errors.Is(err, io.EOF) {
|
|
return fmt.Errorf("error reading tar data: %w", err)
|
|
}
|
|
}
|
|
err = filestore.WFS.WriteFile(ctx, zoneId, fileName, dataBuf)
|
|
if err != nil {
|
|
return fmt.Errorf("error writing to blockfile: %w", err)
|
|
}
|
|
wps.Broker.Publish(wps.WaveEvent{
|
|
Event: wps.Event_BlockFile,
|
|
Scopes: []string{waveobj.MakeORef(waveobj.OType_Block, zoneId).String()},
|
|
Data: &wps.WSFileEventData{
|
|
ZoneId: zoneId,
|
|
FileName: fileName,
|
|
FileOp: wps.FileOp_Invalidate,
|
|
},
|
|
})
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return fmt.Errorf("error copying tar stream: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c WaveClient) Delete(ctx context.Context, conn *connparse.Connection, recursive bool) error {
|
|
zoneId := conn.Host
|
|
if zoneId == "" {
|
|
return fmt.Errorf("zoneid not found in connection")
|
|
}
|
|
schemeAndHost := conn.GetSchemeAndHost() + "/"
|
|
|
|
entries, err := c.ListEntries(ctx, conn, nil)
|
|
if err != nil {
|
|
return fmt.Errorf("error listing blockfiles: %w", err)
|
|
}
|
|
if len(entries) > 0 {
|
|
if !recursive {
|
|
return fmt.Errorf("more than one entry, use recursive flag to delete")
|
|
}
|
|
errs := make([]error, 0)
|
|
for _, entry := range entries {
|
|
fileName := strings.TrimPrefix(entry.Path, schemeAndHost)
|
|
err = filestore.WFS.DeleteFile(ctx, zoneId, fileName)
|
|
if err != nil {
|
|
errs = append(errs, fmt.Errorf("error deleting blockfile %s/%s: %w", zoneId, fileName, err))
|
|
continue
|
|
}
|
|
wps.Broker.Publish(wps.WaveEvent{
|
|
Event: wps.Event_BlockFile,
|
|
Scopes: []string{waveobj.MakeORef(waveobj.OType_Block, zoneId).String()},
|
|
Data: &wps.WSFileEventData{
|
|
ZoneId: zoneId,
|
|
FileName: fileName,
|
|
FileOp: wps.FileOp_Delete,
|
|
},
|
|
})
|
|
}
|
|
if len(errs) > 0 {
|
|
return fmt.Errorf("error deleting blockfiles: %v", errs)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c WaveClient) Join(ctx context.Context, conn *connparse.Connection, parts ...string) (string, error) {
|
|
newPath := path.Join(append([]string{conn.Path}, parts...)...)
|
|
newPath, err := cleanPath(newPath)
|
|
if err != nil {
|
|
return "", fmt.Errorf("error cleaning path: %w", err)
|
|
}
|
|
return newPath, nil
|
|
}
|
|
|
|
func cleanPath(path string) (string, error) {
|
|
if path == "" {
|
|
return "", fmt.Errorf("path is empty")
|
|
}
|
|
if strings.HasPrefix(path, "/") {
|
|
path = path[1:]
|
|
}
|
|
if strings.HasPrefix(path, "~") || strings.HasPrefix(path, ".") || strings.HasPrefix(path, "..") {
|
|
return "", fmt.Errorf("wavefile path cannot start with ~, ., or ..")
|
|
}
|
|
var newParts []string
|
|
for _, part := range strings.Split(path, "/") {
|
|
if part == ".." {
|
|
if len(newParts) > 0 {
|
|
newParts = newParts[:len(newParts)-1]
|
|
}
|
|
} else if part != "." {
|
|
newParts = append(newParts, part)
|
|
}
|
|
}
|
|
return strings.Join(newParts, "/"), nil
|
|
}
|
|
|
|
func (c WaveClient) GetConnectionType() string {
|
|
return connparse.ConnectionTypeWave
|
|
}
|
|
|
|
func getPathPrefix(conn *connparse.Connection) string {
|
|
fullUri := conn.GetFullURI()
|
|
pathPrefix := fullUri
|
|
lastSlash := strings.LastIndex(fullUri, "/")
|
|
if lastSlash > 10 && lastSlash < len(fullUri)-1 {
|
|
pathPrefix = fullUri[:lastSlash+1]
|
|
}
|
|
return pathPrefix
|
|
}
|