2023-01-27 13:37:20 -08:00
|
|
|
// Copyright (c) Tailscale Inc & AUTHORS
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
2022-02-28 13:08:45 -08:00
|
|
|
|
|
|
|
// Package mem provides an in-memory ipn.StateStore implementation.
|
|
|
|
package mem
|
|
|
|
|
|
|
|
import (
|
2023-02-01 16:29:05 -08:00
|
|
|
"bytes"
|
2022-02-28 13:08:45 -08:00
|
|
|
"encoding/json"
|
|
|
|
"sync"
|
|
|
|
|
2024-10-26 09:33:47 -05:00
|
|
|
xmaps "golang.org/x/exp/maps"
|
2022-02-28 13:08:45 -08:00
|
|
|
"tailscale.com/ipn"
|
|
|
|
"tailscale.com/types/logger"
|
2024-10-26 09:33:47 -05:00
|
|
|
"tailscale.com/util/mak"
|
2022-02-28 13:08:45 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
// New returns a new Store.
|
|
|
|
func New(logger.Logf, string) (ipn.StateStore, error) {
|
|
|
|
return new(Store), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Store is an ipn.StateStore that keeps state in memory only.
|
|
|
|
type Store struct {
|
2024-06-26 11:29:53 -04:00
|
|
|
mu sync.Mutex
|
|
|
|
// +checklocks:mu
|
2022-02-28 13:08:45 -08:00
|
|
|
cache map[ipn.StateKey][]byte
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Store) String() string { return "mem.Store" }
|
|
|
|
|
|
|
|
// ReadState implements the StateStore interface.
|
2024-10-26 09:33:47 -05:00
|
|
|
// It returns ipn.ErrStateNotExist if the state does not exist.
|
2022-02-28 13:08:45 -08:00
|
|
|
func (s *Store) ReadState(id ipn.StateKey) ([]byte, error) {
|
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
bs, ok := s.cache[id]
|
|
|
|
if !ok {
|
|
|
|
return nil, ipn.ErrStateNotExist
|
|
|
|
}
|
|
|
|
return bs, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// WriteState implements the StateStore interface.
|
2024-10-26 09:33:47 -05:00
|
|
|
// It never returns an error.
|
2022-02-28 13:08:45 -08:00
|
|
|
func (s *Store) WriteState(id ipn.StateKey, bs []byte) error {
|
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
if s.cache == nil {
|
|
|
|
s.cache = map[ipn.StateKey][]byte{}
|
|
|
|
}
|
2023-02-01 16:29:05 -08:00
|
|
|
s.cache[id] = bytes.Clone(bs)
|
2022-02-28 13:08:45 -08:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-10-26 09:33:47 -05:00
|
|
|
// LoadFromMap loads the in-memory cache from the provided map.
|
|
|
|
// Any existing content is cleared, and the provided map is
|
|
|
|
// copied into the cache.
|
|
|
|
func (s *Store) LoadFromMap(m map[string][]byte) {
|
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
xmaps.Clear(s.cache)
|
|
|
|
for k, v := range m {
|
|
|
|
mak.Set(&s.cache, ipn.StateKey(k), v)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-02-28 13:08:45 -08:00
|
|
|
// LoadFromJSON attempts to unmarshal json content into the
|
|
|
|
// in-memory cache.
|
|
|
|
func (s *Store) LoadFromJSON(data []byte) error {
|
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
return json.Unmarshal(data, &s.cache)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExportToJSON exports the content of the cache to
|
|
|
|
// JSON formatted []byte.
|
|
|
|
func (s *Store) ExportToJSON() ([]byte, error) {
|
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
if len(s.cache) == 0 {
|
|
|
|
// Avoid "null" serialization.
|
|
|
|
return []byte("{}"), nil
|
|
|
|
}
|
|
|
|
return json.MarshalIndent(s.cache, "", " ")
|
|
|
|
}
|