Files
ark/pkg/client-sdk/store/file/store.go
Louis Singer 4da76ec88b New boarding protocol (#279)
* [domain] add reverse boarding inputs in Payment struct

* [tx-builder] support reverse boarding script

* [wallet] add GetTransaction

* [api-spec][application] add reverse boarding support in covenantless

* [config] add reverse boarding config

* [api-spec] add ReverseBoardingAddress RPC

* [domain][application] support empty forfeits txs in EndFinalization events

* [tx-builder] optional connector output in round tx

* [btc-embedded] fix getTx and taproot finalizer

* whitelist ReverseBoardingAddress RPC

* [test] add reverse boarding integration test

* [client] support reverse boarding

* [sdk] support reverse boarding

* [e2e] add sleep time after faucet

* [test] run using bitcoin-core RPC

* [tx-builder] fix GetSweepInput

* [application][tx-builder] support reverse onboarding in covenant

* [cli] support reverse onboarding in covenant CLI

* [test] rework integration tests

* [sdk] remove onchain wallet, replace by onboarding address

* remove old onboarding protocols

* [sdk] Fix RegisterPayment

* [e2e] add more funds to covenant ASP

* [e2e] add sleeping time

* several fixes

* descriptor boarding

* remove boarding delay from info

* [sdk] implement descriptor boarding

* go mod tidy

* fixes and revert error msgs

* move descriptor pkg to common

* add replace in go.mod

* [sdk] fix unit tests

* rename DescriptorInput --> BoardingInput

* genrest in SDK

* remove boarding input from domain

* remove all "reverse boarding"

* rename "onboarding" ==> "boarding"

* remove outdate payment unit test

* use tmpfs docker volument for compose testing files

* several fixes
2024-09-04 19:21:26 +02:00

226 lines
5.9 KiB
Go

package filestore
import (
"context"
"encoding/hex"
"encoding/json"
"fmt"
"os"
"os/user"
"path/filepath"
"strconv"
"strings"
"github.com/ark-network/ark/pkg/client-sdk/internal/utils"
"github.com/ark-network/ark/pkg/client-sdk/store"
"github.com/decred/dcrd/dcrec/secp256k1/v4"
)
const (
filename = "state.json"
)
type storeData struct {
AspUrl string `json:"asp_url"`
AspPubkey string `json:"asp_pubkey"`
WalletType string `json:"wallet_type"`
ClientType string `json:"client_type"`
Network string `json:"network"`
RoundLifetime string `json:"round_lifetime"`
UnilateralExitDelay string `json:"unilateral_exit_delay"`
MinRelayFee string `json:"min_relay_fee"`
BoardingDescriptorTemplate string `json:"boarding_descriptor_template"`
}
func (d storeData) isEmpty() bool {
return d == storeData{}
}
func (d storeData) decode() store.StoreData {
network := utils.NetworkFromString(d.Network)
roundLifetime, _ := strconv.Atoi(d.RoundLifetime)
unilateralExitDelay, _ := strconv.Atoi(d.UnilateralExitDelay)
minRelayFee, _ := strconv.Atoi(d.MinRelayFee)
buf, _ := hex.DecodeString(d.AspPubkey)
aspPubkey, _ := secp256k1.ParsePubKey(buf)
return store.StoreData{
AspUrl: d.AspUrl,
AspPubkey: aspPubkey,
WalletType: d.WalletType,
ClientType: d.ClientType,
Network: network,
RoundLifetime: int64(roundLifetime),
UnilateralExitDelay: int64(unilateralExitDelay),
MinRelayFee: uint64(minRelayFee),
BoardingDescriptorTemplate: d.BoardingDescriptorTemplate,
}
}
func (d storeData) asMap() map[string]string {
return map[string]string{
"asp_url": d.AspUrl,
"asp_pubkey": d.AspPubkey,
"wallet_type": d.WalletType,
"client_type": d.ClientType,
"network": d.Network,
"round_lifetime": d.RoundLifetime,
"unilateral_exit_delay": d.UnilateralExitDelay,
"min_relay_fee": d.MinRelayFee,
"boarding_descriptor_template": d.BoardingDescriptorTemplate,
}
}
type Store struct {
filePath string
}
func NewConfigStore(baseDir string) (store.ConfigStore, error) {
if len(baseDir) <= 0 {
return nil, fmt.Errorf("missing base directory")
}
datadir := cleanAndExpandPath(baseDir)
if err := makeDirectoryIfNotExists(datadir); err != nil {
return nil, fmt.Errorf("failed to initialize datadir: %s", err)
}
filePath := filepath.Join(datadir, filename)
fileStore := &Store{filePath}
if _, err := fileStore.open(); err != nil {
return nil, fmt.Errorf("failed to open store: %s", err)
}
return fileStore, nil
}
func (s *Store) GetType() string {
return store.FileStore
}
func (s *Store) GetDatadir() string {
return filepath.Dir(s.filePath)
}
func (s *Store) AddData(ctx context.Context, data store.StoreData) error {
sd := &storeData{
AspUrl: data.AspUrl,
AspPubkey: hex.EncodeToString(data.AspPubkey.SerializeCompressed()),
WalletType: data.WalletType,
ClientType: data.ClientType,
Network: data.Network.Name,
RoundLifetime: fmt.Sprintf("%d", data.RoundLifetime),
UnilateralExitDelay: fmt.Sprintf("%d", data.UnilateralExitDelay),
MinRelayFee: fmt.Sprintf("%d", data.MinRelayFee),
BoardingDescriptorTemplate: data.BoardingDescriptorTemplate,
}
if err := s.write(sd); err != nil {
return fmt.Errorf("failed to write to store: %s", err)
}
return nil
}
func (s *Store) GetData(_ context.Context) (*store.StoreData, error) {
sd, err := s.open()
if err != nil {
return nil, err
}
if sd.isEmpty() {
return nil, nil
}
data := sd.decode()
return &data, nil
}
func (s *Store) CleanData(ctx context.Context) error {
if err := s.write(&storeData{}); err != nil {
return fmt.Errorf("failed to write to store: %s", err)
}
return nil
}
func (s *Store) open() (*storeData, error) {
file, err := os.ReadFile(s.filePath)
if err != nil {
if !os.IsNotExist(err) {
return nil, fmt.Errorf("failed to open store: %s", err)
}
if err := s.write(&storeData{}); err != nil {
return nil, fmt.Errorf("failed to initialize store: %s", err)
}
return nil, nil
}
data := &storeData{}
if err := json.Unmarshal(file, data); err != nil {
return nil, fmt.Errorf("failed to read file store: %s", err)
}
return data, nil
}
func (s *Store) write(data *storeData) error {
file, err := os.ReadFile(s.filePath)
if err != nil {
if !os.IsNotExist(err) {
return err
}
}
currentData := map[string]string{}
if len(file) > 0 {
if err := json.Unmarshal(file, &currentData); err != nil {
return fmt.Errorf("failed to read file store: %s", err)
}
}
mergedData := merge(currentData, data.asMap())
jsonString, err := json.Marshal(mergedData)
if err != nil {
return err
}
err = os.WriteFile(s.filePath, jsonString, 0755)
if err != nil {
return err
}
return nil
}
func cleanAndExpandPath(path string) string {
// Expand initial ~ to OS specific home directory.
if strings.HasPrefix(path, "~") {
var homeDir string
u, err := user.Current()
if err == nil {
homeDir = u.HomeDir
} else {
homeDir = os.Getenv("HOME")
}
path = strings.Replace(path, "~", homeDir, 1)
}
// NOTE: The os.ExpandEnv doesn't work with Windows-style %VARIABLE%,
// but the variables can still be expanded via POSIX-style $VARIABLE.
return filepath.Clean(os.ExpandEnv(path))
}
func makeDirectoryIfNotExists(path string) error {
if _, err := os.Stat(path); os.IsNotExist(err) {
return os.MkdirAll(path, os.ModeDir|0755)
}
return nil
}
func merge(maps ...map[string]string) map[string]string {
merge := make(map[string]string, 0)
for _, m := range maps {
for k, v := range m {
merge[k] = v
}
}
return merge
}