2024-05-21 20:09:22 +02:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
package wstore
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"log"
|
2024-06-15 23:59:14 +02:00
|
|
|
"path/filepath"
|
2024-05-21 20:09:22 +02:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/jmoiron/sqlx"
|
|
|
|
"github.com/sawka/txwrap"
|
2024-09-05 23:25:45 +02:00
|
|
|
"github.com/wavetermdev/waveterm/pkg/util/migrateutil"
|
|
|
|
"github.com/wavetermdev/waveterm/pkg/wavebase"
|
|
|
|
"github.com/wavetermdev/waveterm/pkg/waveobj"
|
2024-05-22 06:15:11 +02:00
|
|
|
|
2024-09-05 23:25:45 +02:00
|
|
|
dbfs "github.com/wavetermdev/waveterm/db"
|
2024-05-21 20:09:22 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
const WStoreDBName = "waveterm.db"
|
|
|
|
|
|
|
|
type TxWrap = txwrap.TxWrap
|
|
|
|
|
|
|
|
var globalDB *sqlx.DB
|
|
|
|
|
|
|
|
func InitWStore() error {
|
|
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), 2*time.Second)
|
|
|
|
defer cancelFn()
|
|
|
|
var err error
|
|
|
|
globalDB, err = MakeDB(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-05-24 23:08:24 +02:00
|
|
|
err = migrateutil.Migrate("wstore", globalDB.DB, dbfs.WStoreMigrationFS, "migrations-wstore")
|
2024-05-21 20:09:22 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
log.Printf("wstore initialized\n")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetDBName() string {
|
Update data and config paths to match platform defaults (#1047)
Going forward for new installations, config and data files will be
stored at the platform default paths, as defined by
[env-paths](https://www.npmjs.com/package/env-paths).
For backwards compatibility, if the `~/.waveterm` or `WAVETERM_HOME`
directory exists and contains valid data, it will be used. If this check
fails, then `WAVETERM_DATA_HOME` and `WAVETERM_CONFIG_HOME` will be
used. If these are not defined, then `XDG_DATA_HOME` and
`XDG_CONFIG_HOME` will be used. Finally, if none of these are defined,
the [env-paths](https://www.npmjs.com/package/env-paths) defaults will
be used.
As with the existing app, dev instances will write to `waveterm-dev`
directories, while all others will write to `waveterm`.
2024-10-22 18:26:58 +02:00
|
|
|
waveHome := wavebase.GetWaveDataDir()
|
2024-09-05 23:05:42 +02:00
|
|
|
return filepath.Join(waveHome, wavebase.WaveDBDir, WStoreDBName)
|
2024-05-21 20:09:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func MakeDB(ctx context.Context) (*sqlx.DB, error) {
|
|
|
|
dbName := GetDBName()
|
|
|
|
rtn, err := sqlx.Open("sqlite3", fmt.Sprintf("file:%s?mode=rwc&_journal_mode=WAL&_busy_timeout=5000", dbName))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
rtn.DB.SetMaxOpenConns(1)
|
|
|
|
return rtn, nil
|
|
|
|
}
|
|
|
|
|
2024-05-27 22:59:58 +02:00
|
|
|
func WithTx(ctx context.Context, fn func(tx *TxWrap) error) (rtnErr error) {
|
2024-08-20 23:56:48 +02:00
|
|
|
waveobj.ContextUpdatesBeginTx(ctx)
|
2024-05-27 22:59:58 +02:00
|
|
|
defer func() {
|
|
|
|
if rtnErr != nil {
|
2024-08-20 23:56:48 +02:00
|
|
|
waveobj.ContextUpdatesRollbackTx(ctx)
|
2024-05-27 22:59:58 +02:00
|
|
|
} else {
|
2024-08-20 23:56:48 +02:00
|
|
|
waveobj.ContextUpdatesCommitTx(ctx)
|
2024-05-27 22:59:58 +02:00
|
|
|
}
|
|
|
|
}()
|
2024-05-22 06:15:11 +02:00
|
|
|
return txwrap.WithTx(ctx, globalDB, fn)
|
|
|
|
}
|
|
|
|
|
2024-05-27 22:59:58 +02:00
|
|
|
func WithTxRtn[RT any](ctx context.Context, fn func(tx *TxWrap) (RT, error)) (rtnVal RT, rtnErr error) {
|
2024-08-20 23:56:48 +02:00
|
|
|
waveobj.ContextUpdatesBeginTx(ctx)
|
2024-05-27 22:59:58 +02:00
|
|
|
defer func() {
|
|
|
|
if rtnErr != nil {
|
2024-08-20 23:56:48 +02:00
|
|
|
waveobj.ContextUpdatesRollbackTx(ctx)
|
2024-05-27 22:59:58 +02:00
|
|
|
} else {
|
2024-08-20 23:56:48 +02:00
|
|
|
waveobj.ContextUpdatesCommitTx(ctx)
|
2024-05-27 22:59:58 +02:00
|
|
|
}
|
|
|
|
}()
|
2024-05-22 06:15:11 +02:00
|
|
|
return txwrap.WithTxRtn(ctx, globalDB, fn)
|
|
|
|
}
|