2023-10-17 06:31:13 +02:00
|
|
|
// Copyright 2023, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2022-07-07 04:01:00 +02:00
|
|
|
package sstore
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2022-07-13 23:16:08 +02:00
|
|
|
"encoding/base64"
|
2023-07-31 02:16:43 +02:00
|
|
|
"errors"
|
2022-07-13 23:16:08 +02:00
|
|
|
"fmt"
|
2023-07-31 02:16:43 +02:00
|
|
|
"io/fs"
|
2023-03-26 22:21:58 +02:00
|
|
|
"log"
|
2022-09-20 23:15:20 +02:00
|
|
|
"os"
|
|
|
|
"path"
|
2023-12-27 22:11:53 +01:00
|
|
|
"time"
|
2022-07-07 04:01:00 +02:00
|
|
|
|
2023-12-27 22:11:53 +01:00
|
|
|
"github.com/google/uuid"
|
2023-10-16 22:30:10 +02:00
|
|
|
"github.com/wavetermdev/waveterm/waveshell/pkg/cirfile"
|
2024-01-27 01:25:21 +01:00
|
|
|
"github.com/wavetermdev/waveterm/waveshell/pkg/shexec"
|
2023-10-16 22:30:10 +02:00
|
|
|
"github.com/wavetermdev/waveterm/wavesrv/pkg/scbase"
|
2022-07-07 04:01:00 +02:00
|
|
|
)
|
|
|
|
|
2023-07-31 02:16:43 +02:00
|
|
|
func CreateCmdPtyFile(ctx context.Context, screenId string, lineId string, maxSize int64) error {
|
|
|
|
ptyOutFileName, err := scbase.PtyOutFile(screenId, lineId)
|
2022-07-07 04:01:00 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-08-19 22:23:00 +02:00
|
|
|
f, err := cirfile.CreateCirFile(ptyOutFileName, maxSize)
|
2022-08-12 22:59:31 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-08-19 22:23:00 +02:00
|
|
|
return f.Close()
|
|
|
|
}
|
|
|
|
|
2023-07-31 02:16:43 +02:00
|
|
|
func StatCmdPtyFile(ctx context.Context, screenId string, lineId string) (*cirfile.Stat, error) {
|
|
|
|
ptyOutFileName, err := scbase.PtyOutFile(screenId, lineId)
|
2023-02-06 09:30:23 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return cirfile.StatCirFile(ctx, ptyOutFileName)
|
|
|
|
}
|
|
|
|
|
2024-01-27 01:25:21 +01:00
|
|
|
func ClearCmdPtyFile(ctx context.Context, screenId string, lineId string) error {
|
|
|
|
ptyOutFileName, err := scbase.PtyOutFile(screenId, lineId)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
stat, err := cirfile.StatCirFile(ctx, ptyOutFileName)
|
|
|
|
if err != nil && !errors.Is(err, fs.ErrNotExist) {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
os.Remove(ptyOutFileName) // ignore error
|
|
|
|
var maxSize int64 = shexec.DefaultMaxPtySize
|
|
|
|
if stat != nil {
|
|
|
|
maxSize = stat.MaxSize
|
|
|
|
}
|
|
|
|
err = CreateCmdPtyFile(ctx, screenId, lineId, maxSize)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-07-31 02:16:43 +02:00
|
|
|
func AppendToCmdPtyBlob(ctx context.Context, screenId string, lineId string, data []byte, pos int64) (*PtyDataUpdate, error) {
|
2023-03-16 02:12:55 +01:00
|
|
|
if screenId == "" {
|
|
|
|
return nil, fmt.Errorf("cannot append to PtyBlob, screenid is not set")
|
|
|
|
}
|
2022-08-19 22:23:00 +02:00
|
|
|
if pos < 0 {
|
2022-09-05 23:49:23 +02:00
|
|
|
return nil, fmt.Errorf("invalid seek pos '%d' in AppendToCmdPtyBlob", pos)
|
2022-08-19 22:23:00 +02:00
|
|
|
}
|
2023-07-31 02:16:43 +02:00
|
|
|
ptyOutFileName, err := scbase.PtyOutFile(screenId, lineId)
|
2022-08-12 22:59:31 +02:00
|
|
|
if err != nil {
|
2022-09-05 23:49:23 +02:00
|
|
|
return nil, err
|
2022-08-12 22:59:31 +02:00
|
|
|
}
|
2022-08-19 22:23:00 +02:00
|
|
|
f, err := cirfile.OpenCirFile(ptyOutFileName)
|
|
|
|
if err != nil {
|
2022-09-05 23:49:23 +02:00
|
|
|
return nil, err
|
2022-07-08 06:39:25 +02:00
|
|
|
}
|
2022-08-19 22:23:00 +02:00
|
|
|
defer f.Close()
|
|
|
|
err = f.WriteAt(ctx, data, pos)
|
2022-07-07 04:01:00 +02:00
|
|
|
if err != nil {
|
2022-09-05 23:49:23 +02:00
|
|
|
return nil, err
|
2022-07-07 04:01:00 +02:00
|
|
|
}
|
2022-07-13 23:16:08 +02:00
|
|
|
data64 := base64.StdEncoding.EncodeToString(data)
|
|
|
|
update := &PtyDataUpdate{
|
2023-03-16 02:12:55 +01:00
|
|
|
ScreenId: screenId,
|
2023-07-31 02:16:43 +02:00
|
|
|
LineId: lineId,
|
2022-08-19 22:23:00 +02:00
|
|
|
PtyPos: pos,
|
2022-07-13 23:16:08 +02:00
|
|
|
PtyData64: data64,
|
|
|
|
PtyDataLen: int64(len(data)),
|
|
|
|
}
|
2023-07-31 02:16:43 +02:00
|
|
|
err = MaybeInsertPtyPosUpdate(ctx, screenId, lineId)
|
2023-03-26 22:21:58 +02:00
|
|
|
if err != nil {
|
|
|
|
// just log
|
2023-07-31 02:16:43 +02:00
|
|
|
log.Printf("error inserting ptypos update %s/%s: %v\n", screenId, lineId, err)
|
2023-03-26 22:21:58 +02:00
|
|
|
}
|
2022-09-05 23:49:23 +02:00
|
|
|
return update, nil
|
2022-07-07 04:01:00 +02:00
|
|
|
}
|
2022-08-19 22:23:00 +02:00
|
|
|
|
2023-03-26 22:21:58 +02:00
|
|
|
// returns (real-offset, data, err)
|
2023-07-31 02:16:43 +02:00
|
|
|
func ReadFullPtyOutFile(ctx context.Context, screenId string, lineId string) (int64, []byte, error) {
|
|
|
|
ptyOutFileName, err := scbase.PtyOutFile(screenId, lineId)
|
2022-08-19 22:23:00 +02:00
|
|
|
if err != nil {
|
|
|
|
return 0, nil, err
|
|
|
|
}
|
|
|
|
f, err := cirfile.OpenCirFile(ptyOutFileName)
|
|
|
|
if err != nil {
|
|
|
|
return 0, nil, err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
return f.ReadAll(ctx)
|
|
|
|
}
|
2022-09-20 23:15:20 +02:00
|
|
|
|
2023-03-26 22:21:58 +02:00
|
|
|
// returns (real-offset, data, err)
|
2023-07-31 02:16:43 +02:00
|
|
|
func ReadPtyOutFile(ctx context.Context, screenId string, lineId string, offset int64, maxSize int64) (int64, []byte, error) {
|
|
|
|
ptyOutFileName, err := scbase.PtyOutFile(screenId, lineId)
|
2023-03-26 22:21:58 +02:00
|
|
|
if err != nil {
|
|
|
|
return 0, nil, err
|
|
|
|
}
|
|
|
|
f, err := cirfile.OpenCirFile(ptyOutFileName)
|
|
|
|
if err != nil {
|
|
|
|
return 0, nil, err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
return f.ReadAtWithMax(ctx, offset, maxSize)
|
|
|
|
}
|
|
|
|
|
2022-09-20 23:15:20 +02:00
|
|
|
type SessionDiskSizeType struct {
|
|
|
|
NumFiles int
|
|
|
|
TotalSize int64
|
|
|
|
ErrorCount int
|
2022-12-27 04:06:46 +01:00
|
|
|
Location string
|
2022-09-20 23:15:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func directorySize(dirName string) (SessionDiskSizeType, error) {
|
|
|
|
var rtn SessionDiskSizeType
|
2022-12-27 04:06:46 +01:00
|
|
|
rtn.Location = dirName
|
2022-09-20 23:15:20 +02:00
|
|
|
entries, err := os.ReadDir(dirName)
|
|
|
|
if err != nil {
|
|
|
|
return rtn, err
|
|
|
|
}
|
|
|
|
for _, entry := range entries {
|
|
|
|
if entry.IsDir() {
|
|
|
|
rtn.ErrorCount++
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
finfo, err := entry.Info()
|
|
|
|
if err != nil {
|
|
|
|
rtn.ErrorCount++
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
rtn.NumFiles++
|
|
|
|
rtn.TotalSize += finfo.Size()
|
|
|
|
}
|
|
|
|
return rtn, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func SessionDiskSize(sessionId string) (SessionDiskSizeType, error) {
|
2022-12-27 03:49:45 +01:00
|
|
|
sessionDir, err := scbase.EnsureSessionDir(sessionId)
|
2022-09-20 23:15:20 +02:00
|
|
|
if err != nil {
|
|
|
|
return SessionDiskSizeType{}, err
|
|
|
|
}
|
|
|
|
return directorySize(sessionDir)
|
|
|
|
}
|
|
|
|
|
|
|
|
func FullSessionDiskSize() (map[string]SessionDiskSizeType, error) {
|
2022-12-27 03:49:45 +01:00
|
|
|
sdir := scbase.GetSessionsDir()
|
2022-09-20 23:15:20 +02:00
|
|
|
entries, err := os.ReadDir(sdir)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
rtn := make(map[string]SessionDiskSizeType)
|
|
|
|
for _, entry := range entries {
|
|
|
|
if !entry.IsDir() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
name := entry.Name()
|
|
|
|
_, err = uuid.Parse(name)
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
diskSize, err := directorySize(path.Join(sdir, name))
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
rtn[name] = diskSize
|
|
|
|
}
|
|
|
|
return rtn, nil
|
|
|
|
}
|
2022-12-22 02:45:40 +01:00
|
|
|
|
2023-07-31 02:16:43 +02:00
|
|
|
func DeletePtyOutFile(ctx context.Context, screenId string, lineId string) error {
|
|
|
|
ptyOutFileName, err := scbase.PtyOutFile(screenId, lineId)
|
2022-12-22 02:45:40 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-07-31 02:16:43 +02:00
|
|
|
err = os.Remove(ptyOutFileName)
|
|
|
|
if errors.Is(err, fs.ErrNotExist) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return err
|
2022-12-22 02:45:40 +01:00
|
|
|
}
|
2022-12-27 01:09:21 +01:00
|
|
|
|
2023-12-27 22:11:53 +01:00
|
|
|
func GoDeleteScreenDirs(screenIds ...string) {
|
|
|
|
go func() {
|
|
|
|
for _, screenId := range screenIds {
|
|
|
|
deleteScreenDirMakeCtx(screenId)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
func deleteScreenDirMakeCtx(screenId string) {
|
|
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), time.Minute)
|
|
|
|
defer cancelFn()
|
|
|
|
err := DeleteScreenDir(ctx, screenId)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("error deleting screendir %s: %v\n", screenId, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-21 03:20:57 +01:00
|
|
|
func DeleteScreenDir(ctx context.Context, screenId string) error {
|
|
|
|
screenDir, err := scbase.EnsureScreenDir(screenId)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("error getting screendir: %w", err)
|
|
|
|
}
|
2023-12-27 22:11:53 +01:00
|
|
|
log.Printf("delete screen dir, remove-all %s\n", screenDir)
|
2023-03-21 03:20:57 +01:00
|
|
|
return os.RemoveAll(screenDir)
|
|
|
|
}
|